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Préface 


# .NET est le langage de programmation phare de Microsoft. Il a été développé 
dans le but de pouvoir créer facilement divers types d'applications en tirant le 
meilleur des produits et technologies Microsoft. 


Les créateurs du langage se sont inspirés des langages existants en s’attachant à re- 
tenir le meilleur de chacun d’eux. Aussi n’est-ce pas étonnant de retrouver un typage 
fort, une approche orientée objet et une syntaxe rappellant à la fois celle du C++ et 
du Java. C# .NET est apparu en 2000 et depuis, ne cesse d’évoluer au rythme des 
différentes versions du Framework .NET. Le couple C# et Framework .NET englobe 
les dernières avancées des langages de programmation (Generic, Lambda, Inférence de 
type, Linq...). Ces améliorations, fortement inspirées des langages dits fonctionnels, 
font de C# un des langages les plus modernes et aboutis, sans que jamais la produc- 
tivité et la solidité du code ne soient compromis. Aujourd’hui, C#.NET est de plus 
en plus utilisé dans le monde professionnel. Sa puissance et son interopérabilité avec 
les produits et technologies Microsoft font de lui un langage sûr et pérenne. Ce lan- 
gage présente en outre l'intérêt de ne pas être propriétaire puisque ses spécifications 
permettent de voir apparaître des initiatives (comme par exemple Mono), le code C4 
pouvant ainsi tourner sur des distributions Linux. Il est possible de développer toutes 
sortes d'applications : jeux, applications de gestion, interfaces tactiles, XAML ou appli- 
cations pour téléphones. N'oublions pas le monde embarqué avec le Micro Framework 
.NET ainsi que le Web avec ASP.NET. Bref, C# est un langage tout terrain, ouvrant 
une gamme de possibles unique sur la plateforme Microsoft. 


Ce livre se veut simple et facile d’accès. Il allie les connaissances de Nicolas Hilaire, 
spécialiste de ces technologies et MVP Microsoft !, avec celles des créateurs du Site 
du Zéro, réputés depuis de nombreuses années pour leur approche pédagogique et 
accessible à tous les débutants. Ce livre est donc tout indiqué pour ceux qui veulent se 
former facilement à la programmation C# .NET. 


Éric Mittelette 
Responsable des relations techniques avec les développeurs chez Microsoft France 


1. Most Valuable Professional, expert en technologies Microsoft. 
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Avant-propos 


uand j'ai commencé la programmation, j'avais dix ans et un Atari ST possédant 

un interpréteur GFA Basic. Mes parents m’avaient acheté un livre contenant des 

listings à recopier et à exécuter. Si mes souvenirs ne me trahissent pas, il s’agis- 
sait pour la plupart d'applications permettant de gérer le contenu de son frigo ou de sa 
cave à vins. Quelques petits jeux très simples et peu graphiques venaient agrémenter 
le lot. Pour faire fonctionner ces programmes, il fallait tout recopier à la main (ou 
plutôt au clavier), généralement quelques centaines de lignes de code. Régulièrement, 
cela ne fonctionnait pas car je faisais une erreur de copie, inversant des parenthèses ou 
oubliant des mots. À part vérifier tout le listing ligne par ligne, je n’avais plus qu’à 
passer au listing suivant ! Parfois, mes efforts étaient récompensés même si je ne com- 
prenais strictement rien à ce que je recopiais. Je me rappelle d’un superbe labyrinthe 
en 3 dimensions, quoique mes souvenirs lui rendent certainement un hommage plus en 
couleur qu’il ne le méritait ! Ces listings remplis de mots magiques m'ont donné envie 
de comprendre comment cela fonctionnait. J’ai donc pris mon courage à dix doigts 
et tenté de créer mes propres programmes en isolant les parties qui me paraissaient 
simples. Afficher « Bonjour comment allez-vous » et pouvoir « discuter » avec l’ordi- 
nateur grâce à un algorithme de mon cru ont été un de mes premiers souvenirs de 
programme réussi. 


À cette époque reculée, il n’existait pas de moyen d'apprendre facilement la program- 
mation. Il n’y avait pas internet... eh oui, cette époque a existé! Durant mon adoles- 
cence j’ai continué mon apprentissage en essayant différents langages, comme le C++ 
ou l’assembleur, le turbo pascal et autres joyeusetés. La plupart étaient inaccessibles, 
notamment le C++. Quelques livres en bibliothèque ont fini dans la mienne mais ils 
étaient tous bien incompréhensibles... je me souviens même d’un livre qui promettait 
de pouvoir créer un jeu « facilement ». Cela ne devait pas être si facile que ça vu mon 
air hébété après la lecture du livre! Cela manquait d’un Site du Zéro où tout est ex- 
pliqué de zéro pour les personnes, comme j’ai pu l’être, curieuses de se lancer dans le 
monde magique du développement. 
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CHAPITRE 0. AVANT-PROPOS 


On parle du CÆ# ? 


J'y viens! C’est dans cette optique que j’ai commencé à écrire. Pouvoir partager mes 
connaissances souvent durement acquises et aider ceux qui ont du mal à se lancer. Et 
c’est vrai que ce n’est pas facile, malgré toute la bonne volonté du monde. Sans une 
méthodologie simple et des explications claires, il n’est pas aisé de se lancer sans se sentir 
perdu. C’est là où j'espère pouvoir faire quelque chose à travers la collection des Livres 
du Zéro. Après tous mes essais de jeunesse, mes études et mon entrée dans le monde du 
travail, j'ai acquis une certaine expérience des différents langages de programmation. 
J’ai pris goût à l'écriture en commençant à rédiger des articles avec mon langage préféré 
de l’époque, le C++. Aujourd’hui, c’est le C{# qui occupe cette place prestigieuse dans 
mon classement ultra-personnel des langages de programmation! C’est donc l’occasion 
de pouvoir mettre à profit cette volonté de partage de connaissances et ce goût pour 
la rédaction, dans un ouvrage permettant d'apprendre le C# et qui est destiné aux 
débutants. 


Qu’allez-vous apprendre en lisant ce livre ? 


Nous allons apprendre le langage de programmation C+# de façon progressive au cours 
de cet ouvrage, composé des parties suivantes : 


1. Les rudiments du langage C# : nous commencerons par découvrir les bases 
du langage CÆ. Nous partons vraiment des bases : comment est construite une 
application informatique ? Quels logiciels dois-je installer ? Quelles sont les ins- 
tructions de base du C#? Nous allons découvrir tout cela au cours de cette 
première partie qui permettra de poser les briques de nos premières applications. 


2. Un peu plus loin avec le C< : dans cette partie, nous allons continuer à appro- 
fondir nos connaissances avec le C#. Nous découvrirons les premières interactions 
avec l'utilisateur de nos programmes. Comment lire simplement une saisie cla- 
vier ? Comment lire le contenu de la ligne de commande ? Nous découvrirons cela, 
avec en complément des TP pour nous entraîner. 


3. Le C#, un langage orienté objet : ici, les choses sérieuses commencent et nous 
allons voir ce qu’est la programmation orientée objet et comment le C# répond 
à ce genre de programmation. Chapitre un peu plus avancé où vous découvrirez 
toute la puissance du langage et où vous vous rendrez compte de l'étendue des 
possibilités du C;! 


4. C#Æ Avancé : forts de nos connaissances acquises précédemment, nous étudie- 
rons des points plus avancés dans cet ultime chapitre. Nous verrons comment 
accéder efficacement aux données grâce à LINQ et comment utiliser une base de 
données avec Entity Framework. Nous verrons également d’autres aspects per- 
mettant d’être encore plus efficaces avec vos développements. 
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COMMENT LIRE CE LIVRE ? 


À la fin de cet ouvrage, vous aurez acquis toutes les bases vous permettant de vous lan- 
cer sans appréhension dans le monde du développement d’applications professionnelles 
avec le C#. Vous découvrirez en bonus un aperçu des différentes applications que l’on 
peut réaliser avec le C#. 


Comment lire ce livre ? 


Esprit du livre 


Oui, oui, vous avez bien lu, ce livre est pour les débutants. Pas besoin d’avoir fait du 
développement auparavant pour pouvoir lire cet ouvrage! Je vais donc faire de mon 
mieux pour détailler au maximum mes explications, c’est promis. Bien sûr, il y en a 
peut-être parmi vous qui ont déjà fait du C, du C++, du Java... Évidemment, si vous 
avez déjà fait du développement informatique, ce sera plus facile pour vous. Attention 
néanmoins de ne pas aller trop vite : le C# ressemble à d’autres langages mais il a 
quand même ses spécificités! Nous allons découvrir ensemble de nombreuses choses 
en apprenant à développer en C#. Il y aura bien entendu des TP pour vous faire 
pratiquer, afin que vous puissiez vous rendre compte de ce que vous êtes capables de 
faire après avoir lu plusieurs chapitres plus théoriques. Néanmoins, je veux que vous 
soyez actifs ! Ne vous contentez pas de lire passivement mes explications, même lorsque 
les chapitres sont plutôt théoriques! Testez les codes et les manipulations au fur et à 
mesure. Essayez les petites idées que vous avez pour améliorer ou adapter légèrement 
le code. Sortez un peu des sentiers battus du tutoriel : cela vous fera pratiquer et 
vous permettra de découvrir rapidement si vous avez compris ou non le chapitre. Pas 
d'inquiétude, si jamais vous bloquez sur quoi que ce soit qui n’est pas expliqué dans ce 
cours, la communauté qui sillonne les forums du Site du Zéro saura vous apporter son 
aide précieuse. 


Suivez l’ordre des chapitres 


Lisez ce livre comme on lit un roman. Il a été conçu pour cela. 


Contrairement à beaucoup de livres techniques où il est courant de lire en diagonale et 
de sauter certains chapitres, il est ici très fortement recommandé de suivre l’ordre du 
cours, à moins que vous ne soyez déjà un peu expérimentés. 


Utilisez les codes web! 


Afin de tirer parti du Site du Zéro dont ce livre est issu, celui-ci vous propose ce qu’on 
appelle des « codes web ». Ce sont des codes à six chiffres à saisir sur une page du Site 
du Zéro pour être automatiquement redirigé vers un site web sans avoir à en recopier 
l’adresse. 


CHAPITRE 0. AVANT-PROPOS 


Pour utiliser les codes web, rendez-vous sur la page suivante ? : 
http://www.siteduzero.com/codeweb.html 


Un formulaire vous invite à rentrer votre code web. Faites un premier essai avec le code 
ci-dessous : 


Tester le code web 
Code web : 123456 


Ces codes web ont deux intérêts : 


— ils vous redirigent vers les sites web présentés tout au long du cours, vous permettant 
ainsi d’obtenir les logiciels dans leur toute dernière version ; 

— ils vous permettent de télécharger les codes sources inclus dans ce livre, ce qui vous 
évitera d’avoir à recopier certains programmes un peu longs. 


Ce système de redirection nous permet de tenir à jour le livre que vous avez entre les 
mains sans que vous ayez besoin d’acheter systématiquement chaque nouvelle édition. 
Si un site web change d’adresse, nous modifierons la redirection mais le code web à 
utiliser restera le même. Si un site web disparaît, nous vous redirigerons vers une page 
du Site du Zéro expliquant ce qui s’est passé et vous proposant une alternative. 


En clair, c’est un moyen de nous assurer de la pérennité de cet ouvrage sans que vous 
ayez à faire quoi que ce soit ! 


Ce livre est issu du Site du Zéro 


Cet ouvrage reprend le cours C# du Site du Zéro dans une édition revue et corrigée, 
augmentée de nouveaux chapitres plus avancés et des notes de bas de page. 


Il reprend les éléments qui ont fait le succès des cours du site, à savoir leur approche 
progressive et pédagogique, leur ton décontracté, ainsi que les TP vous permettant de 
pratiquer de façon autonome. 


Ce livre s’adresse donc à toute personne désireuse d'apprendre les bases de la program- 
mation en C#, que ce soit : 

— par curiosité ; 

— par intérêt personnel ; 

— par besoin professionnel. 
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Première partie 


Les rudiments du langage C- 


Introduction au C 


Difficulté : 1 


ans ce tout premier chapitre, nous allons découvrir ce qu'est le C#, son histoire 
et son rapport avec le framework .NET. D'ailleurs, vous ne savez pas ce qu'est un 
framework ? Ce n'est pas grave, tout ceci sera expliqué | 


Nous verrons dans ce chapitre ce que sont les applications informatiques et comment des 
langages de programmation évolués comme le C# nous permettent de réaliser de telles 
applications. 


Et ce n'est que le début... alors ouvrez grands vos yeux, chaussez vos lunettes et explorons 
ce monde merveilleux ! 
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Qu'est-ce que le C7? 


Le C# est un langage de programmation créé par Microsoft en 2002. 


ensemble de mots qui permettent de construire des applications informatiques. 


Q Un langage de programmation est un ensemble d'instructions, c'est-à-dire un 


Ces applications informatiques peuvent être de beaucoup de sortes, par exemple une 
application Windows, comme un logiciel de traitement de texte, une calculatrice ou 
encore un jeu de cartes. On les appelle également des clients lourds. Il est égale- 
ment possible de développer des applications web, comme un site d’e-commerce, un 
intranet, etc. Nous pourrons accéder à ces applications grâce à un navigateur inter- 
net que l’on appelle un client léger. Toujours grâce à un navigateur internet, nous 
pourrons développer des clients riches. Ce sont des applications qui se rapprochent 
d’une application Windows mais qui fonctionnent dans un navigateur. Bien d’autres 
types d'applications peuvent être écrites avec le C#, citons encore le développement 
d'applications mobiles sous Windows phone 7, de jeux ou encore de web services... 
Nous verrons un peu plus en détail à la fin de cet ouvrage comment réaliser de telles 
applications ! 


Le C# est un langage dont la syntaxe ressemble un peu au C++ ou au Java qui sont 
d’autres langages de programmation très populaires. Le CÆ est le langage phare de 
Microsoft. Il fait partie d’un ensemble plus important : il est en fait une brique de ce 
qu'on appelle le & Framework .NET ». Gardons encore un peu de suspens sur ce 
qu'est le framework .NET, nous découvrirons ce que c’est un peu plus loin dans ce 
livre. 


Comment sont créées les applications informatiques ? 


Une application informatique : qu'est-ce que c’est ? 


Comme vous le savez, votre ordinateur exécute des applications informatiques pour 
effectuer des tâches. Ce sont des logiciels comme : 


— un traitement de texte; 

— un navigateur internet ; 

— un jeu vidéo; 

— etc. 

Votre ordinateur ne peut exécuter ces applications informatiques que si elles sont écrites 
dans le seul langage qu’il comprend, le binaire. Techniquement, le binaire est représenté 
par une suite de 0 et de 1, comme vous pouvez le voir à la figure 1.1. 


Il n’est bien sûr pas raisonnablement possible de réaliser une grosse application en 
binaire, c’est pour ça qu'il existe des langages de programmation qui permettent de 
simplifier l'écriture d’une application informatique. 
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COMMENT SONT CRÉÉES LES APPLICATIONS INFORMATIQUES ? 


FIGURE 1.1 - Le langage binaire est composé de 0 et de 1 


Comment créer des programmes « simplement » ? 


Je vais vous expliquer rapidement le principe de fonctionnement des langages dits 
traditionnels, comme le C et le C++, puis je vous présenterai le fonctionnement du 
C#. Comme le C# est plus récent, il a été possible d'améliorer son fonctionnement par 
rapport au C et au C++ comme nous allons le voir. 


Langages traditionnels : la compilation 


Avec des langages traditionnels comme le C et le C++, on écrit des instructions sim- 
plifiées, lisibles par un humain comme : 


1 | printf("Bonjour"); 


Ce n’est pas vraiment du français, mais c’est quand même beaucoup plus simple que 
le binaire et on comprend globalement avec cet exemple que l’on va afficher le mot 
« Bonjour ». 


Bien entendu, l'ordinateur ne comprend pas ces instructions. Lui, il veut du binaire, du 
vrai! Pour obtenir du binaire à partir d’un code écrit en C ou C++, on doit effectuer 
ce qu’on appelle une compilation. Le compilateur est un programme qui traduit le 
code source en binaire exécutable. Ce n’est pas clair ? Observez donc la figure 1.2! 


printf ("Bonjour"); 


tC 


Ecriture dans un langage de Compilation Programme en binaire 
programmation (traduction) Lisible par l'ordinateur 
(C, C++...) 


FIGURE 1.2 - Compilation traditionnelle d’un programme en binaire 


Cette méthode est efficace et a fait ses preuves. De nombreuses personnes développent 
toujours en C et C++ aujourd’hui. Néanmoins, ces langages ont aussi un certain nombre 
de défauts dus à leur ancienneté. Par exemple, un programme compilé (binaire) ne 
fonctionne que sur la plateforme pour laquelle il a été compilé. Cela veut dire que si 
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vous compilez sous Windows, vous obtenez un programme qui fonctionne sous Windows 
uniquement (et sur un type de processeur particulier). Impossible de le faire tourner 
sous Mac OS X ou Linux, à moins de le recompiler sous ces systèmes d’exploitation et 
d'effectuer au passage quelques modifications (voir figure 1.3). 


re" 


Programme pour 


C ilati 
ompilation Windows 
= 
Compilation Programme pour Linux 


os 


c dati Programme pour 
ompilation Mac OS X 


FIGURE 1.3 —- Compilations multiples pour cibler toutes les plateformes 


Les programmes binaires ont ce défaut : ils ne fonctionnent que pour un type de ma- 
chine. Pour les développeurs qui écrivent le code, c’est assez fastidieux à gérer. 


Langages récents : le code managé 


Les langages récents, comme le C# et le Java, résolvent ce problème de compatibilité 
tout en ajoutant au langage de nombreuses fonctionnalités appréciables, ce qui permet 
de réaliser des programmes beaucoup plus efficacement. 


La compilation en C# ne donne pas un programme binaire, contrairement au € et au 
C++. Le code C# est en fait transformé dans un langage intermédiaire (appelé CIL 
ou MSIL) que l’on peut ensuite distribuer à tout le monde. Ce code, bien sûr, n’est pas 
exécutable lui-même, car l’ordinateur ne comprend que le binaire. 


Regardez bien la figure 1.4 pour comprendre comment cela fonctionne. 


Le code en langage intermédiaire (CIL) correspond au programme que vous allez distri- 
buer. Sous Windows, il prend l’apparence d’un .exe comme les programmes habituels, 
mais il ne contient en revanche pas de binaire. 
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Exécution du programme dans un système d'exploitation 


—6 “= -+$-5 


Code en langage C# Compilation Code en langage Compilation du Programme binaire 
intermédiaire (CIL) Common Language 
Runtime (CLR) 


FIGURE 1.4 - Compilation C# et exécution du CIL par le CLR 


Lorsqu'on exécute le programme CIL, celui-ci est lu par un autre programme (une 
machine à analyser les programmes, appelée CLR) qui le compile cette fois en vrai 
programme binaire. Cette fois, le programme peut s’exécuter, ouf! 


Q Ça complique bien les choses quand même! Est-ce bien utile ? 


Cela offre beaucoup de souplesse au développeur. Le code en langage intermédiaire 
(CIL) peut être distribué à tout le monde. Il suffit d’avoir installé la machine CLR sur 
son ordinateur, qui peut alors lire les programmes en C# et les compiler « à la volée » 
en binaire. Avantage : le programme est toujours adapté à l’ordinateur sur lequel il 
tourne. 


Le CLR vérifie aussi la sécurité du code; ainsi en € du code mal pensé (par 
exemple une mauvaise utilisation des pointeurs) peut entraîner des problèmes 
pour votre PC, ce que vous risquez beaucoup moins avec le C#. De plus, le 
CLR dispose du JIT debugger qui permet de lancer Visual Studio si une erreur 
survient dans un programme .NET pour voir ce qui a causé cette erreur. On 
parle de code « managé ». 


Cette complexité ralentit légèrement la vitesse d'exécution des programmes (par rap- 
port au C ou au C++), mais la différence est aujourd’hui vraiment négligeable par 
rapport aux gains que cela apporte. 


Donc, en théorie, il est possible d'utiliser n'importe quelle application compilée en 
langage intermédiaire à partir du moment où il y a une implémentation du CLR dispo- 
nible. En réalité, il n’y a que sous Windows qu'il existe une implémentation complète 
du CLR. Il existe cependant une implémentation partielle sous Linux : Mono. Cela 
veut dire que si votre programme utilise des fonctionnalités qui ne sont pas couvertes 
par Mono, il ne fonctionnera pas. 


En conclusion, dans la pratique, le .NET est totalement exploitable sous Windows et 
ailleurs, non. 
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Exécutables ou assemblages ? 


J’ai dit juste au dessus que le C# était compilé en langage intermédiaire et qu’on le 
retrouve sous la forme d’un .exe comme les programmes habituels. 


À C'est vrai! (Je ne mens jamais!) 


Par contre, c’est un peu incomplet. 


Il est possible de créer des programmes (.exe) qui pourront directement être exécutés 
par le CLR, mais il est également possible de créer des bibliothèques sous la forme d’un 
fichier possédant l’extension .d11. 


On appelle ces deux formes de programme des assemblages, mais on utilise globalement 
toujours le mot anglais assembly. 


— Les fichiers .exe sont des assemblys de processus. 
— Les fichiers .d11 sont des assemblys de bibliothèques. 


Concrètement, cela signifie que le fichier .exe servira à lancer une application et qu’une 
dll pourra être partagée entre plusieurs applications .exe afin de réutiliser du code 
déjà écrit. 


Nous verrons un peu plus loin comment ceci est possible. 


tendance à croire que le mot assembly sert à désigner uniquement les biblio- 
thèques dont l'extension est .d11. 


Q@ Il est à noter qu'un raccourci est souvent fait avec le terme assembly. On a 


Qu'est-ce que le framework .NET ? 


J'ai commencé à vous parler du C# qui était une brique du framework .NET. Il est 
temps d’en savoir un peu plus sur ce fameux framework. 


Commençons par le commencement : comment cela se prononce ? 


DOTTE NETTE ou encore POINT NETTE. Je vous accorde que le nom est bizarre. ..! 
Surtout que le nom peut être trompeur. Avec l’omniprésence d’internet, son abréviation 
(net) ou encore des noms de domaines (.net), on pourrait penser que le framework .NET 
est un truc dédié à internet. Que nenni! 

Nous allons donc préciser un peu ce qu'est le framework .NET pour éviter les ambi- 
guités. 


© Première chose à savoir, qu'est-ce qu'un framework ? 


QU'EST-CE QUE LE FRAMEWORK .NET ? 


Pour simplifier, on peut dire qu’un framework est une espèce de grosse boîte à fonc- 
tionnalités qui va nous permettre de réaliser des applications informatiques de toutes 
sortes. 


PIN En fait, c'est la combinaison de ce framework et du langage de programmation 


CH qui va nous permettre de réaliser ces applications informatiques. 


Le framework .NET a été créé par Microsoft en 2002, en même temps que le C#, qui est 
principalement dédié à la réalisation d'applications fonctionnant dans des environne- 
ments Microsoft. Nous pourrons par exemple réaliser des programmes qui fonctionnent 
sous Windows, ou bien des sites web ou encore des applications qui fonctionnent sur 
téléphone mobile, etc. 


Disons que la réalisation d’une application informatique, c’est un peu comme un chan- 
tier (je ne dis pas ça parce qu'il y a toujours du retard, même si c’est vrai!). Il est 
possible de construire différentes choses, comme une maison, une piscine, une terrasse, 
etc. Pour réaliser ces constructions, nous allons avoir besoin de matériaux, comme des 
briques, de la ferraille, etc. Certains matériaux sont communs à toutes les construc- 
tions (fer, vis...) et d’autres sont spécifiques à certains domaines (pour construire une 
piscine, je vais avoir besoin d’un liner par exemple). 


On peut voir le framework .NET comme ces matériaux ; c’est un ensemble de compo- 
sants que l’on devra assembler pour réaliser notre application. Certains sont spécifiques 
pour la réalisation d'applications web, d’autres pour la réalisation d’applications Win- 
dows, etc. 


Pour réaliser un chantier, nous allons avoir besoin d'outils afin de manipuler les ma- 
tériaux. Qui envisagerait de tourner une vis avec les doigts ou de poser des parpaings 
sans les coller avec du mortier ? C’est la même chose pour une application informatique, 
pour assembler notre application, nous allons utiliser un langage de programmation : 


le C4. 


À l'heure où j'écris ces lignes, le C# est en version 4 ainsi que le framework .NET. 
Ce sont des versions stables et utilisées par beaucoup de personnes. Chaque version 
intermédiaire a apporté son lot d’évolutions. Le framework .NET et le C# sont en 
perpétuelle évolution, preuve de la dynamique apportée par Microsoft. 


C’est tout ce qu'il y a à savoir pour l'instant ! Nous reviendrons un peu plus en détail 
sur le framework .NET dans les chapitres suivants. Pour l’heure, il est important de 
retenir que c’est grâce au langage de programmation CÆ et grâce aux composants du 
framework .NET que nous allons pouvoir développer des applications informatiques. 


En résumé 


— Le C# est un langage de programmation permettant d'utiliser le framework .NET. 
C’est le langage phare de Microsoft. 
— Le framework .NET est une énorme boîte à fonctionnalités permettant la création 
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d'applications. 

— Le C# permet de développer des applications de toutes sortes, exécutables par le 
CLR qui traduit le MSIL en binaire. 

— Il est possible de créer des assemblys de deux sortes : des assemblys de processus 
exécutables par le CLR et des assemblys de bibliothèques. 
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Créer un projet avec Visual C# 2010 
Express 


Difficulté : 16: 


ans ce chapitre nous allons faire nos premiers pas avec le C#. Nous allons dans un 

premier temps installer et découvrir les outils qui nous seront nécessaires pour réaliser 

des applications informatiques avec le C#. Nous verrons comment démarrer avec ces 
outils et à la fin de ce chapitre, nous serons capables de créer un petit programme qui 
affiche du texte simple et nous aurons commencé à nous familiariser avec l'environnement 
de développement. 


Il faut bien commencer par les bases, mais vous verrez comme cela peut être gratifiant 
d'arriver enfin à faire un petit quelque chose. Allez, c'est parti! 
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Que faut-il pour démarrer ? 


J'espère vous avoir donné envie de démarrer l’apprentissage du C#, cependant, il nous 
manque encore quelque chose pour pouvoir sereinement attaquer cet apprentissage. 
Bien sûr, vous allez avoir besoin d’un ordinateur, mais a priori, vous l’avez déjà! S'il 
n'est pas sous Windows, mais sous Linux, vous pouvez utiliser Mono qui va permettre 
d'utiliser le CÆ sous Linux. Il est téléchargeable via le code web suivant : 


Télécharger Mono ) 


Code web : 566528 


Cependant, Mono n’est pas aussi complet que le C# et le framework .NET sous Win- 
dows. En l’utilisant vous risquez de passer à côté de certaines parties de ce livre. 


Pour reprendre la métaphore du chantier, on peut dire qu’il va également nous manquer 
un chef de chantier. Il n’est pas forcément nécessaire en théorie, mais dans la pratique 
il se révèle indispensable pour mener à bien son chantier. 


Ce chef de chantier c’est en fait l’outil de développement. Il va nous fournir les 
outils pour orchestrer nos développements. 


C’est entre autres : 


— un puissant éditeur, qui va nous servir à créer des fichiers contenant des instruc- 
tions en langage CÆ#; 

— un compilateur, qui va servir à transformer ces fichiers en une suite d’instructions 
compréhensibles par l’ordinateur, comme nous l’avons déjà vu; 

— un environnement d’exécution, qui va permettre de faire les actions informa- 
tiques correspondantes (afficher une phrase, réagir au clic de la souris, etc.) ; c’est le 
CLR dont on a déjà parlé. 


Enfin, nous aurons besoin d’une base de données. Nous y reviendrons plus en détail 
ultérieurement, mais la base de données est un endroit où seront stockées les données de 
notre application. C’est un élément indispensable à mesure que l’application grandit. 


Installer Visual C7 2010 Express 


Nous avons donc besoin de notre chef de chantier, l’outil de développement. C’est 
un logiciel qui va nous permettre de créer des applications et qui va nous fournir les 
outils pour orchestrer nos développements. La gamme de Microsoft est riche en outils 
professionnels de qualité pour le développement, notamment grâce à Visual Studio. 


Integrated Development Environment ce qui signifie « Environnement de dé- 


@ Notez que cet outil de développement se nomme également un IDE pour 
veloppement intégré ». 


Nous aurons recours au terme IDE régulièrement. 


Pour apprendre et commencer à découvrir l’environnement de développement, Micro- 
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soft propose gratuitement Visual Studio dans sa version Express. C’est une version 
allégée de l’environnement de développement qui permet de faire plein de choses, mais 
avec des outils en moins par rapport aux versions payantes. Rassurez-vous, ces ver- 
sions gratuites sont très fournies et permettent de faire tout ce dont on a besoin pour 
apprendre le C# et suivre ce cours. 


Pour réaliser des applications d'envergure, il pourra cependant être judicieux d’in- 
vestir dans l’outil complet et ainsi bénéficier de fonctionnalités complémentaires qui 
permettent d'améliorer, de faciliter et d’industrialiser les développements. 


Pour développer en C# gratuitement et créer des applications Windows, nous allons 
avoir besoin de Microsoft Visual C# 2010 Express que vous pouvez télécharger 
en suivant ce code web : 


Visual C# 2010 Express ) 


Code web : 389622 


Pour résumer : 


— Visual Studio est la version payante de l’outil de développement. 
— Microsoft Visual C# 2010 Express est une version allégée et gratuite de Visual 
Studio, dédiée au développement en C#. Exactement ce qu’il nous faut ! 


Cliquez sur Visual C# 2010 Express et choisissez la langue qui vous convient. Puis 
cliquez sur Téléchargez (voir les figures 2.1 et 2.2). 


#2 Télécharger les versions E... * V&h 


+ C ft © msdnsmicrosoft.com/fr-fr/express/a2975050 
Avant de procéder à l'installation, vous devez désinstaller toute version bêta, CTP ou Tech Preview précédente de SQL Server 2008 R2, 


Visual Studio 2010 et NET Framework 3.5. 
Étape 2 : Téléchargement et installation 
En fonction des types de projets que vous voulez créer, choisissez l'outil le plus approprié pour télécharger diverses éditions Express. 


>. SQL Server 2008 R2 


Express 


© Visual Studio 2010 


Express for Windows Phone 


« Libérez votre créativité: créez vos applications 
Windows Phone grâce aux technologies Silverlight et 
XNA.» 


FIGURE 2.1 — Page de téléchargement de Visual C# Express 2010 
Une fois l’exécutable téléchargé, il ne reste plus qu’à le lancer et l’installation démarre. 
Cliquez sur Suivant pour démarrer l'installation, ainsi qu’indiqué à la figure 2.3. 


Vous devez ensuite lire la licence d'utilisation du logiciel et l’accepter pour pouvoir 
continuer l'installation. 


Une application sans données, c’est plutôt rare. C’est un peu comme un site d’e- 
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#3 Merci de télécharger Mic... * 
< © # © msanmicrosoft.com/fr-fr/99699226 


Bienvenue dans le d'installation , 
vdi © Visual C# 2010 
Express 


Bienvenue dans l'Assistant Installation de Microsoft Visual C# 2010 Express. Microsoft Visual C# 
2010 Express est un outil de développement simple, amusant et fade à utiliser destiné aux 
programmeurs C# qui souhaitent créer des applications Windows Forms et Windows Presentation 
Foundation (WPF), des bibliothèques de dasses et aussi des applications consoles. Cet Assistant 
va vous quider tout au long de l'installation, Si le produit requiert des composants qui ne sont pas 
encore installés sur l'ordinateur, vous pourrez également les installer. 


Aider à améliorer l'installation 
Vous pouvez envoyer des informations sur votre installation à Microsoft. Pour 
participer, activez la case à cocher ci-dessous. 
[[] Oui, envoyer des informations relatives à mon installation à Microsoft Corporation. 
{i) Pour plus d'informations, cliquez sur Déclaration de confidentialité 


FIGURE 2.3 - Écran d'accueil du programme d'installation de Visual C# Express 
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commerce sans produits, un traitement de texte sans fichiers ou le site du zéro sans 
tutoriel. On risque de vite s’ennuyer ! 


Heureusement, le programme d’installation nous propose d'installer « Microsoft SQL 
Server 2008 Express Service Pack 1 ». Microsoft propose en version gratuite un serveur 
de base de données allégé. Il va nous permettre de créer facilement une base de données 
et de l’utiliser depuis nos applications en C#. 


ultérieur mais une base de données est un énorme endroit où sont stockées 
les données de notre application. 


@ Nous l'avons déjà évoqué et nous y reviendrons plus en détail dans un chapitre 


Nous avons également évoqué dans l'introduction qu’il était possible de réaliser des 
applications qui ressemblent à des applications Windows mais dans un navigateur, que 
nous avons appelé client riche. Silverlight va justement permettre de créer ce genre 
d'application. 


Cochez donc tout pour installer Silverlight et Sal Server et cliquez sur Suivant (voir 
figure 2.4). 


ptions d'installati ar 
slt anei ©@ Visual C# 2010 
Express 


Sélectionnez les produits optionnels à installer : 


rosoft Silverlight (Taille du téléchargement : 4.7 MB) 
Microsoft Silverlight est un plug-in de navigateur qui permet des expériences Web riches. En 
installant Silverlight vous acceptez le contrat de licence Silverlight. 


EN osoft SQL Server 2008 Express Service Pack 1 (x86) (Taille du : 
| sc : MB; Î 


SQL Server Express s'intègre à Visual Studio añn de fournir les fonctionnalités de bases de 
données dient et serveur. 


(i) Pour plus d'informations, consultez le fichier Readme. 


FIGURE 2.4 - Installer Silverlight et Microsoft Sql Server Express 2008 


Cliquez ensuite sur Installer en changeant éventuellement le dossier d'installation. 
Une fois l'installation terminée cliquez sur Quitter. 


À l'heure où j'écris ces lignes, il existe un service pack pour Visual Studio, le « Service 
pack 1 ». C’est un ensemble de corrections qui permettent d'améliorer la stabilité du 
logiciel. Je vous invite à télécharger et installer ce pack, disponible via le code web 
suivant : 
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Service pack 1 
Code web : 974305 


Vous voilà avec votre copie de Visual C# Express qui va vous permettre de créer des 
programmes en C# gratuitement et facilement. L'installation de l’outil de développe- 
ment est terminée. 


Notez que, bien que gratuite, vous aurez besoin d'enregistrer votre copie de 
Visual C# Express avant 30 jours. C'est une opération rapide et nécessitant 
un compte Windows Live. Après cela, vous pourrez utiliser Visual C# Express 
sans retenue. 


En résumé, nous avons installé un outil de développement, Visual C# 2010 dans sa 
version Express et une base de données, SQL Server Express 2008. Nous avons tous les 
outils nécessaires et nous allons pouvoir démarrer (enfin !) l’apprentissage et la pratique 


du C#. 


Démarrer Visual C7 2010 Express 


Nous allons vérifier que l'installation de Visual C# Express a bien fonctionné. Et pour 
ce faire, nous allons le démarrer et commencer à prendre en main ce formidable outil 
de développement. 


Il vous semblera sûrement très complexe au début mais vous allez voir, si vous suivez 
ce livre pas à pas, vous allez apprendre les fonctionnalités indispensables. Elles seront 
illustrées par des copies d'écran vous permettant de plus facilement vous y retrouver. 
À force d'utiliser Visual C# Express, vous verrez que vous vous sentirez de plus en 
plus à l’aise et peut-être oserez-vous aller fouiller dans les menus ? 


Commencez par démarrer Visual C7 2010 Express. Le logiciel s’ouvre sur une page de 
démarrage (voir la figure 2.5). 


Fichier Edition Affichage Déboguer Outils Fenêtre ? 


Ford | 4 3m -e-| >| _ _______-|IF+#8. 


+ Explorateur de solutions 


Microsote= 
CO Visual C# 2010 Express 


Prise en main Dernières informations 
F Nouveau projet... 


= Bienvenue Apprendre Mise à niveau 
5] Ouvrir un projet... 


G a ==} Bienvenue dans Visual 
Projets récents = C# 2010 Express 

La tradition se poursuit. Visual 
C# 2010 Express permet aux 
développeurs de créer 
rapidement des applications 
interactives passionnantes 
pour Windows. Grâce au 


FIGURE 2.5 — Premier démarrage de Visual C# Express 
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CRÉER UN PROJET 


Les deux zones entourées de rouge permettent respectivement de créer un nouveau 
projet et d'accéder aux anciens projets déjà créés. Dans ce deuxième cas, comme je 
viens d'installer le logiciel, la liste est vide. 


Créer un projet 


Commençons par créer un nouveau projet en cliquant dans la zone rouge. Cette com- 
mande est également accessible via le menu Fichier > Nouveau > Projet. 


Un projet va contenir les éléments de ce que l’on souhaite réaliser. Cela peut être par 
exemple une application web, une application Windows, etc. 


Le projet est aussi un container de fichiers et notamment dans notre cas de fichiers en 
langage C# qui vont permettre de construire ce que l’on souhaite réaliser. Le projet 
est en fait représenté par un fichier dont l’extension est .csproj. Son contenu décrit les 
paramètres de configuration correspondant à ce que l’on souhaite réaliser et les fichiers 
qui composent le projet. 


Créons donc un nouveau projet. La fenêtre de création de nouveau projet s’ouvre et 
nous avons plusieurs possibilités de choix. Nous allons dans un premier temps aller 
dans Visual C# pour choisir de créer une Application console, comme indiqué à la 
figure 2.6. 


Nouveau projet 
Trier par: [Par défaut » Rechercher Modèles installés 
Modèles installés 
: # 
visual C4 | 1zc#) Application Windows Forms Visual C# per ee 
ual C# = Projet de création d'une application en 
ligne de commande 
Modèles en ligne LE! Application WPF Visual C# 


LA Application console Visual C# 
er: Bibliothèque de classes Visual C# 
==} 


c#| Application de navigateur WPF Visual C# 


Projet vide Visual C# 
ch] Proj 


MaPremiereApplication 


FIGURE 2.6 — Créer une application console avec Visual C# Express 
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À noter que si vous n'avez installé que Visual CH Express, vous aurez la même 
fenêtre que moi. Si vous disposez de la version payante de Visual Studio, alors 
la fenêtre sera certainement plus garnie. De même, il y aura plus de choses 
si vous avez installé d'autres outils de la gamme Express. 


Ce que nous faisons ici, c’est utiliser ce qu’on appelle un « modèle »de création de 
projet, plus couramment appelé par son équivalent anglais template. 


Si vous naviguez à l’intérieur des différents modèles, vous pourrez constater que Visual 
C# nous propose des modèles de projets plus ou moins compliqués. Ces modèles sont 
très utiles pour démarrer un projet car toute la configuration du projet est déjà faite. 
Le nombre de modèles peut être différent en fonction de votre version de Visual Studio 
ou du nombre de versions Express installées. 


L'application console est la forme de projet pouvant produire une application exé- 
cutable la plus simple. Elle permet de réaliser un programme qui va s’exécuter dans la 
console noire qui ressemble à une fenêtre ms-dos, pour les dinosaures comme moi qui 
ont connu cette époque... À noter que les projets de type « Bibliothèque de classes » 
permettent de générer des assemblys de bibliothèques (.d11). 


Dans cette console, nous allons notamment pouvoir afficher du texte simple. Ce type 
de projet est parfait pour démarrer l’apprentissage du C# car il y a seulement besoin 
de savoir comment afficher du texte pour commencer. 


En bas de la fenêtre de création de projet, nous avons la possibilité de choisir un nom 
pour le projet, ici ConsoleApplicationi. Changeons le nom de notre application en, 
par exemple, MaPremiereApplication, dans la zone correspondante. 


Cliquons sur OK pour valider la création de notre projet. 


Visual C# Express crée alors pour nous les fichiers composant une application console 
vide, qui utilise le C4 comme langage et que nous avons nommée « MaPremiereAppli- 
cation ». 


Avant toute chose, nous allons enregistrer le projet. Allons dans le menu Fichier > 
Enregistrer (ou utiliser le raccourci bien connu “e (s). 


Visual C; Express nous ouvre la fenêtre d’enregistrement de projet (voir figure 2.7). 


+ [4% Main(stringl] args) ge 

Eusing System; ÆT CA solution MaPremiereApplication’ (1 projet) 
| using System.Collections.Generic; + | 4 A MaPremiereApplication 

| using System. Ling; a Properties 

using System.Text; sa Références 


5 MaPremiereApplication.Program 


Einanespace MaPremiereApplication 4 Programes 
1€ 


CAUsers\Nico\documents\visual studio 2010\Projects 


Emplacement : 


Nom de solution: MaPremiereApplication Pi Créer le répertoire pour la solution 


FIGURE 2.7 — Enregistrer la solution 


Nous pouvons donner un nom, préciser un emplacement où nous souhaitons que les 
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ANALYSE DE L'ENVIRONNEMENT DE DÉVELOPPEMENT ET DU CODE 
GÉNÉRÉ 


fichiers soient enregistrés et un nom de solution. Une case pré-cochée nous propose de 
créer un répertoire pour la solution. C’est ce que nous allons faire ! Cliquons maintenant 
sur Enregistrer. 


nom de la solution sont à renseigner au moment où l'on crée le projet. Une 


! Pour les versions payantes de Visual Studio, le choix de l'emplacement et le 
différence subtile ! 


Analyse de l’environnement de développement et du 
code généré 


Si l’on se rend dans l’emplacement renseigné (ici c:\users\nico\documents\visual 
studio 2010\Projects), nous pouvons constater que Visual C7 Express a créé un 
répertoire MaPremiereApplication : c’est le fameux répertoire pour la solution qu’il 
nous a proposé de créer. 


Dans ce répertoire, nous remarquons notamment un fichier MaPremiereApplication.sln. 


C’est ce qu’on appelle le fichier de solution; il s’agit juste d’un container de projets 
qui va nous permettre de visualiser nos projets dans Visual C# Express. 


En l'occurrence, nous n’avons pour l'instant qu’un projet dans la solution : l’appli- 
cation MaPremiereApplication, que nous pouvons retrouver dans le sous-répertoire 
MaPremiereApplication et qui contient notamment le fichier de projet qui s'intitule : 
MaPremiereApplication.csproji. 


Q Le fichier décrivant un projet écrit en C# est préfixé par csproji. 


Il y à encore un fichier digne d'intérêt dans ce répertoire, il s’agit du fichier Program. cs. 
Les fichiers dont l’extension est .cs contiennent du code C#, c’est dans ce fichier que 
nous allons commencer à taper nos premières lignes de code... 


de programmation est appelé le « code source ». Par extension, le « code » 
correspond à des instructions écrites dans un langage de programmation. 


Q@ L'ensemble des fichiers contenant des instructions écrites dans un langage 


Retournons dans l'interface de Visual C# Express, et détaillons un peu les différentes 
zones ! Pour suivre mes explications, reportez-vous à la figure 2.8. 


— La zone verte numéro 1 contient les différents fichiers ouverts sous la forme d’un on- 
glet. On voit que, par défaut, Visual CÆ# nous a créé et ouvert le fichier Program.cs. 
— Dans la zone rouge numéro 2, c’est l’éditeur de code. Il affiche le contenu du fichier 
ouvert. Nous voyons des mots que nous ne comprenons pas encore. C’est du code qui 
a été automatiquement généré par Visual C#. Nous pouvons observer que les mots 
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LA MaîremiereAp 


Pere 


plication - Microsoft Visual C# 2010 Express (Administrateur) Eos | 


> 2 
using C3 Solution MaPremiereApplicetiont (1 projet) 
using System.Collections.Generic; 
using System. Ling; 
using System. Text; &a Références 
2] Programs 
inemespace MaPremiereapplication 9 Program: 

cless Progren 


static void Main(string() args) 


MaPremiereApplcation Propriétés du projet + 


CHE 
Fichier projet MaPremierepphcetion.c 


Colonne 


Fichier projet 
Nom du fichuer contenant la build, la 
<orfigurabon et d'autres infornetions sur le pro... 


FIGURE 2.8 - L'IDE de Visual C# Express 


sont de différentes couleurs. En effet, l’éditeur Visual C# Express possède ce qu’on 
appelle une coloration syntaxique, c’est-à-dire que certains mots-clés sont colorés 
d’une couleur différente en fonction de leur signification ou de leur contexte afin de 
nous permettre de nous y retrouver plus facilement. Comme dans ce livre! 

— La zone numéro 3 en violet est l’explorateur de solutions. C’est ici que l’on voit 
le contenu de la solution sur laquelle nous travaillons en ce moment. En l’occur- 
rence, il s’agit de la solution MaPremiereApplication qui contient un unique projet 
MaPremiereApplication. Ce projet contient plusieurs sous éléments : 


1. Properties : contient des propriétés de l’application, on ne s’en occupe pas 
pour l'instant ; 


2. Références : contient les références de l’application, on ne s’en occupe pas non 
plus pour l'instant ; 


3. Program. cs est le fichier qui a été généré par Visual CÆ et qui contient le code 
C;. Il nous intéresse fortement ! 


— La zone numéro 4 en brun est la zone contenant les propriétés de ce sur quoi nous 
travaillons en ce moment. Ici, nous avons le curseur positionné sur le projet, il n’y 
a pas beaucoup d’informations excepté le nom du fichier de projet. Nous aurons 
l’occasion de revenir sur cette fenêtre plus tard. 

— La zone numéro 5 en jaune n’est pas affichée au premier lancement, elle contient 
la liste des erreurs, des avertissements et des messages de notre application. Nous 
verrons comment l’afficher un peu plus bas. 

— La zone numéro 6 en noir est la barre d'outils, elle possède plusieurs boutons que 
nous pourrons utiliser, notamment pour exécuter notre application. 
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ÉCRIRE DU TEXTE DANS NOTRE APPLICATION 


Écrire du texte dans notre application 


Allons donc dans la zone 2 réservée à l’édition de notre fichier Program.cs qui est le 
fichier contenant le code C# de notre application. 


Les mots présents dans cette zone sont ce qu’on appelle des instructions de langage. 
Elles vont nous permettre d'écrire notre programme. Nous reviendrons plus loin sur ce 
que veulent dire les instructions qui ont été générées par Visual C#. Pour l’instant, 
rajoutons simplement l'instruction suivante après l’accolade ouvrante : 


1 | Console.WriteLine("Hello World !!"); 


de manière à avoir : 


1| static void Main(string[] args) 

21 { 

3 Console.WriteLine("Hello World !!"); 
al } 


Nous venons d'écrire une instruction qui va afficher la phrase « Hello World !! ». Pour 
l’instant vous avez juste besoin de savoir ça. Nous étudierons ultérieurement à quoi 
cela correspond exactement. 


L’exécution du projet 


Ça y est! Nous avons écrit notre premier code qui affiche un message très populaire. 
Mais pour le moment, ça ne fait rien... On veut voir ce que ça donne! 


Comme je vous comprends ! 


La première chose à faire est de transformer le langage C# que nous venons d'écrire 
en programme exécutable. Cette phase s’appelle la « génération de la solution » sous 
Visual C#. On l’appelle souvent la « compilation » ou en anglais le build . 


Allez dans le menu Déboguer et cliquez sur Générer la solution (voir figure 2.9). 


Visual CÆ lance alors la génération de la solution et on voit dans la barre des tâches 
en bas à gauche qu’il travaille jusqu’à nous indiquer que la génération a réussi, ainsi 
que vous pouvez le voir sur la figure 2.10. 

Si nous allons dans le répertoire contenant la solution, nous pouvons voir dans le 
répertoire MaPremiereApplication\MaPremiereApplication\bin\Release qu'il y à 
deux fichiers : 


— MaPremiereApplication.exe 
— MaPremiereApplication.pdb 


Le premier est le fichier exécutable, possédant l’extension .exe, qui est le résultat du 
processus de génération. Il s’agit bien de notre application. 


Le second est un fichier particulier qu’il n’est pas utile de connaître pour l'instant, nous 
allons l’ignorer ! 
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Pas à pas détaillé 


Æ Pas à pas principal F10 
E pas princip: Dre 


Eusing System; Basculer le point d'arrêt F9 
using System.Collecti Fenét ; 
using System.Linqg; 
using System.Text; Effacer tous les DataTips 

; | Exporter les DataTips.… 

Elnamespace MaPremiereA : 

{ Importer les DataTips.. 
E class Program Options et paramètres... 
E static void Main(string[] args) 

{ 
Console.WriteLine("Bonjour !!"}); 
3: 
LD à} 
; 


FIGURE 2.9 — Générer la solution 


La génération a réuss 


FIGURE 2.10 — Génération réussie ! 
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L'EXÉCUTION DU PROJET 


Exécutons notre application en lançant le fichier exécutable depuis l’explorateur de 
fichiers. Déception, nous voyons à peine un truc noirâtre qui s’affiche et qui se referme 
immédiatement. Que s’est-il passé ? En fait, l’application s’est lancée, a affiché notre 
message et s’est terminée immédiatement. Et tout ça un brin trop rapidement... ça ne 
va pas être pratique tout ça. Heureusement, Visual C# Express arrive à la rescousse. 
Retournons dans notre IDE préféré. Nous allons ajouter un bouton dans la barre d’ou- 
tils. J'avoue ne pas comprendre pourquoi ce bouton est manquant dans l'installation 
par défaut. Nous allons remédier à ce problème en cliquant sur la petite flèche qui est à 
côté de la barre d’outils, tout à droite, et qui nous ouvre le menu déroulant permettant 
d'ajouter ou supprimer des boutons. Cliquez sur Personnaliser (voir figure 2.11). 


Fa En Monge Pr Dogue Den Ou ne ? 


EE IE #a@m|9-@-|p 10280 JS +eE 
PCR x Æ1= 2 ; [y] 1 Nouveau projet. Ctri+Maj+N | LEE 
[| É] Ajouter un élément |__| 
RP LB Ouvrir un fichier. Ctri+O = 
Eusing System; F2 Mi Enregistrer MaPremiereApplication.sin Ctri+S _— 
using Den ass -Gen! |V] ff Enregistrertout Ctri+Maj+S 
as. nr # at a 
| opier tri 
Einamespace MaPremiereApplicat] FI È “es CteV 
ni jéisés Progres |V| 4 Annuler Ctri+Z 
[LV © Rétablir Ctri+V 
E static void Main(str] [7], b Démarrerle débogage F5 
| : Console .WriteLin! LV | ÆŸ Rechercher dans les fichiers Ctri+Maj+F 
} F2 Rechercher 
} } |LV. €] &plorateur de solutions Ctr+W, S 
PF Fenêtre Propriétés Ctri+W, P 
|Y| 3 Boîte à outils CtrieW, X 
F2 Page de démarrage 
LV ù 


FIGURE 2.11 -— Personnalisation de la barre d’outils 


Cliquez sur Ajouter une commande, comme indiqué à la figure 2.12. 


Allez dans la catégorie Déboguer et choisissez Exécuter sans débogage puis cliquez 
sur OK (voir figure 2.13). 


Enfin, fermez la fenêtre. Comme vous pouvez l’observer sur la figure 2.14, vous avez 
désormais un nouveau bouton dans la barre d'outils! 


Si vous avez la flemme d’ajouter ce bouton, vous pouvez aussi utiliser le raccourci 
+ ou bien cliquer sur ce nouveau bouton pour exécuter l’application depuis 
Visual C4. La console s'ouvre enfin, nous délivrant le message tant attendu (voir figure 
2:15): 


Le message est désormais visible car Visual C# nous demande d’appuyer sur une 
touche pour que l’application se termine, ce qui nous laisse donc le temps d’apprécier 
l’exécution de notre superbe programme. 


La console noire que nous voyons sur la figure 2.15 est le résultat de l’exécution de 
notre programme. Si vous n'avez pas créé ce programme en même temps que moi, vous 
pouvez voir à quoi ressemble cette console. 
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pre, 


Choisissez le menu ou la barre d'outils à réorganiser : 


© Barre de menus: | Barre de menus 


© Barre d'outils : Standard 
© Menu contextuel : | Menus contextuels de l'éditeur 


Contrôles : 


Enregistrer les éléments sélectionnés 
Enregistrer tout 


Couper 
Copier 
Coller 


Annuler plusieurs niveaux 
Refaire plusieurs niveaux 


FIGURE 2.12 — Ajouter une commande à la barre d'outils 


Sélectionnez la commande à ajouter et cliquez sur OK. 
Catégories : Commandes : 

? 

Affichage 

Concepteur de requêtes 


Importer les DataTips 
Imprimer 


Instruction du langage 
WU Interrompretout 
Ligne 


a. 


FIGURE 2.13 — Ajouter la commande d’exécution sans débogage 


L’EXÉCUTION DU PROJET 


Le] MaPremiereApplication - Microsoft \ 
él Edition Affichage Projet ! 


Jr SA mi) 4 = 
RENE EER 

PS Programs" X 

œ 

Er #31 laPremiereApplication.Progran 
m 


Dincine Eucktame 


FIGURE 2.14 - Exécuter le projet sans débogage 


F5 


D | 
EX C:\Windows\system32\cmd.exe l nil 1 À — 


Hello World ?? 
Appuyez sur une touche pour continuer... 


FIGURE 2.15 - Notre première application C# dans la console 


CHAPITRE 2. CRÉER UN PROJET AVEC VISUAL C# 2010 EXPRESS 


Dans la suite du livre, nous utiliserons la syntaxe suivante pour remplacer l’image de 
console du dessus : 


Hello World !! 


Waouh, ça y est, notre première application en C1! 


Je suis fier de nous, mais nous n’allons pas en rester là, nous sommes désormais fin 
parés pour apprendre le C;}! 


En résumé 


— Visual C# Express est l’outil de développement gratuit de Microsoft permettant de 
démarrer avec le C#. 

— Visual Studio est l’outil de développement payant de Microsoft permettant d’être 
efficace dans le développement d’applications .NET. 

— Microsoft SQL Server Express est le moteur de base de données utilisable facilement 
avec Visual C# Express. 

— L'environnement de développement nous permet de créer du code C# qui sera 
contenu dans des projets, qui peuvent être réunis dans une solution. 
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La syntaxe générale du C# 


Difficulté : 1 


ous allons aborder ici la syntaxe générale du C# dans le cadre d'une application 

console. Nous allons utiliser très souvent l'instruction Console.WriteLine("..."); 

que nous avons vue au chapitre précédent et qui est dédiée à l'affichage sur la console. 
C'est une instruction qui va s'avérer très pratique pour notre apprentissage car nous pour- 
rons avoir une représentation visuelle de ce que nous allons apprendre. 


Préparez-vous, nous plongeons petit à petit dans l'univers du C#. Dans ce chapitre, nous 
allons nous attaquer à la syntaxe générale du CH et nous serons capables de reconnaître 
les lignes de code et de quoi elles se composent. 
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Ecrire une ligne de code 


Les lignes de code écrites avec le langage de développement C7 doivent s’écrire dans 
des fichiers dont l’extension est .cs. Nous avons vu dans le chapitre précédent que 
nous avons écrit dans le fichier Program. cs qui est le fichier qui a été généré par Visual 
C# lors de la création du projet. Nous y avons notamment ajouté une instruction 
permettant d'afficher du texte. Les lignes de code C# se lisent et s’écrivent de haut en 
bas et de gauche à droite, comme un livre normal. Aussi, une instruction écrite avant 
une autre sera en général exécutée avant celle-ci. 


Attention, chaque ligne de code doit être correcte d'un point de vue syn- 
taxique sinon le compilateur ne saura pas la traduire en langage exécutable. 


Par exemple, si à la fin de mon instruction, je retire le point-virgule ou si j’orthographie 
mal le mot WriteLine, Visual C# Express me signalera qu’il y a un problème, comme 
vous pouvez le voir à la figure 3.1. 


LÆ] MaPremiereApplication - Microsoft Visual C# 2010 Express (Administrateur) 

Fichier Edition Affichage Refactoriser Projet Générer Déboguer Données Outils Fenêtre ? 
igradéddl4aml9-e-g-Slr 
PL Rs 


|D#G48Q.- 


Main 
= g System; 

ing System.Collections.Generic; 

using System.Linqg; 

using System.Text; 


MUTETTEN 


Einamespace MaPremiereApplication 
ES] class Program 
Œ] static void Main(string[] args) 


| Console.WritLine("Bonjour !!"). 


| | 


Liste d'erreurs 


(D 2eme | 0 metisemnt | Demers | 


Description 


@ 1 ‘System.Console' ne contient pas de définition pour ‘WritLine 


O2 :attendu 
FIGURE 3.1 - Mise en valeur des erreurs de compilation 


Visual C7 Express met en valeur un manque au niveau de la fin de l'instruction et il 
me souligne également le mot « WritLine ». Dans la fenêtre du bas, il m'indique qu’il 
y a deux erreurs et me donne des précisions sur celles-ci avec éventuellement des pistes 
pour résoudre ces erreurs. 
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LE CARACTÈRE DE TERMINAISON DE LIGNE 


Si je tente de lancer mon application (avec le raccourci (Ctr1}+(F5), Visual C# Ex- 
press va tenter de compiler et d’exécuter l’application. Ceci n'étant pas possible, il 
m'affichera un message indiquant qu’il y a des erreurs (voir figure 3.2). 


Microsoft Visual C# 2010 Express 


Des erreurs de build se sont produites. Voulez-vous continuer et 
1 exécuter la dernière build réussie ? 


Ne plus afficher cette boîte de dialogue 


FIGURE 3.2 —- Compilation impossible suite à des erreurs 


Ce sont des erreurs de compilation qu'il va falloir résoudre si l’on souhaite que l’appli- 
cation console puisse s’exécuter. 


Nous allons voir dans les chapitres suivants comment écrire correctement des instruc- 
tions en C#. Mais il est important de noter à l’heure actuelle que le C# est sensible 
à la casse, ce qui veut dire que les majuscules comptent ! Aïnsi le mot WriteLine et le 
mot WriTEline sont deux mots bien distincts et peuvent potentiellement représenter 
deux instructions différentes. Ici, le deuxième mot est incorrect car il n’existe pas. 


Rappelez-vous bien que la casse est déterminante pour que l'application puisse 
compiler. 


Le caractère de terminaison de ligne 


En général, une instruction en code C# s’écrit sur une ligne et se termine par un 
point-virgule. Ainsi, l'instruction que nous avons vue plus haut : 


1 | Console.WriteLine ("Hello World !!"); 


se termine au niveau du point-virgule. Il aurait été possible de remplacer le code écrit : 


1] class Program 

2| € 

3 static void Main(string[]l args) 

4 { 

5 Console.WriteLine("Hello World !!"); 
6 } 

7|} 

par : 


1| class Program {static void Main(string[]l args) {Console. 
WriteLine("Hello World !!");}} 
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ou encore : 

1] class Program 

21 À 

3| static void Main(string[] args) 
4 { 

5 Console 

6 

7 

8| .-WriteLine("Hello World !!" 
9 

10 >; 

11 } 

12|} 


En général, pour que le code soit le plus lisible possible, on écrit une instruction par 
ligne et on indente le code de façon à ce que les blocs soient lisibles. 


Un bloc de code est délimité par des accolades : { et }. Nous y reviendrons 
plus tard. 


Indenter signifie que chaque ligne de code qui fait partie d'un même bloc de 
code commence avec le même retrait sur l'éditeur. Ce sont soit des tabula- 
tions, soit des espaces qui permettent de faire ce retrait. 


Visual C# Express nous aide pour faire correctement cette indentation quand nous 
écrivons du code. Il peut également remettre toute la page en forme avec la combinaison 
de touche : [Ctr1)+{K)+{Ctr1)+{p} 


Décortiquons à présent cette ligne de code : 


1 | Console.WriteLine ("Hello World !!"); 


Pour simplifier, nous dirons que nous appelons la méthode WriteLine qui permet 
d'écrire une chaîne de caractères sur la console. 


Une méthode représente une fonctionnalité, écrite avec du code, qui est utili- 
sable par d'autres bouts de code (par exemple, calculer la racine carrée d’un 
nombre ou afficher du texte. .….). 


L’instruction "Hello World !!" représente une chaîne de caractères et est passée en 
paramètre de la méthode Console .WriteLine à l’aide des parenthèses. La chaîne de 
caractères est délimitée par les guillemets. Enfin, le point-virgule permet d’indiquer 
que l'instruction est terminée et qu’on peut enchaîner sur la suivante. 


Certains points ne sont peut-être pas encore tout à fait clairs, comme ce qu’est vraiment 
une méthode, ou comment utiliser des chaînes de caractères, mais ne vous inquiétez 
pas, nous allons y revenir plus en détail dans les chapitres suivants et découvrir au fur 
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et à mesure les arcanes du C#. 


Les commentaires 


Pour faciliter la compréhension du code ou pour se rappeler un point précis, il est 
possible de mettre des commentaires dans son code. Les commentaires sont ignorés par 
le compilateur et n’ont qu’une valeur informative pour le développeur. Dans un fichier 
de code C# (.cs), on peut écrire des commentaires de deux façons différentes : 


— en commençant son commentaire par /* et en le terminant par */ ce qui permet 
d'écrire un commentaire sur plusieurs lignes ; 

— en utilisant // et tout ce qui se trouve après sur la même ligne est alors un commen- 
taire. 


Visual C# Express colore les commentaires en vert pour faciliter leur identification. 


1| /* permet d'afficher du texte 


2 sur la console */ 
3| Console.WriteLine("Hello World !!"); // ne pas oublier le point 
virgule 


À noter qu'on peut commenter plusieurs lignes de code avec le raccourci 
clavier |Ctr1|+1|K]+/|Ctr1l)+1|C|et décommenter plusieurs lignes de code 


avec le raccourci clavier [Ctr1)+{[K)+/{[Ctrl)+|u) 


La complétion automatique 


Visual C# Express est un formidable outil qui nous facilite à tout moment la tâche, 
notamment grâce à la complétion automatique. La complétion automatique est le 
fait de proposer de compléter automatiquement ce que nous sommes en train d'écrire 
en se basant sur ce que nous avons le droit de faire. Par exemple, si vous avez cherché 
à écrire l'instruction : 


1 | Console.WriteLine("Hello World !!"); 


vous avez pu constater que lors de l’appui sur la touche (él Visual C# Express nous 
affiche une fenêtre avec tout ce qui commence par « C », ainsi que vous pouvez l’observer 
à la figure 3.3. 


Au fur et à mesure de la saisie, il affine les propositions pour se positionner sur la 
plus pertinente. Il est possible de valider la proposition en appuyant sur la touche 
(Entrée ). Non seulement cela nous économise des appuis de touches, paresseux comme 
nous sommes, mais cela nous permet également de vérifier la syntaxe de ce que nous 
écrivons et d’obtenir également une mini-aide sur ce que nous essayons d’utiliser (voir 


figure 3.4). 
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Lusing System. Text; 
=inamespace MaPremiereApplication 
{ 
E class Program 
{ 
E static void Main(string[] args) 
{ 
£ 
É } Ÿ = break # 
[+ #4 Buffer a 
. 1 byte 
(® Byte 
= case Exception levée lors 
(= catch 
0x 5 1 char 
ace 1 Char , prenne) 
EP  — + Î 


FIGURE 3.3 — La complétion automatique facilite l’écriture du code 


static void Main(string[] args) 


conl 

F |A class . 
|%g CLSCompliantAttribute 
lag Comparer<> 
(5 Comparison<> 


class System.Console 


Représente les flux d'entrée, de sortie et d'erreur standard pour les applications console.Cette classe ne 
peut pas être héritée. 


(4 ConsoleCancelEventArgs 
à ConsoleCancelEventHandier 
&P ConsoleColor 


# ConsoleKey * 


FIGURE 3.4 — La complétion automatique s’affine au fur et à mesure de la saisie 
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Ainsi, fini les fautes de frappe qui entraînent une erreur de compilation ou les listes de 
mots-clés dont il faut absolument retenir l’orthographe! 


De la même façon, une fois que vous avez fini de saisir « Console » vous allez saisir le 
point «.» et Visual C# Express va nous proposer toute une série d'instructions en 
rapport avec le début de l'instruction (voir figure 3.5). 


static void Main(string[] args) 


Console. 
} © SetWindowSize = 
EF Title 
 TreatControlCAsnput 
Æ WindowHeight 
SF WindowLeft 
Là WindowTop 
7 WindowWidth = 
|=@ Write a ——————— —— — —_—— = ——— 
=@ |WriteLine void Console.WriteLine(string format, params object[] arg) (+ 18 surcharge(s)) 


[um 


7e G Écrit dans le flux de sortie standard la représentation textuelle du tableau d'objets spécifié, suivie de la 
1 0 avertissements | marque de fin de ligne active, à l'aide des informations de format spécifiées, 
m 
e g = Exceptions : 
assignation, un appel, un incrément, un décrément e System.IO.I0Exception 
struction System.ArgumentNullException 
System.FormatException 


FIGURE 3.5 — La complétion automatique 


Nous pourrons ainsi facilement finir de saisir WriteLine et ceci sans erreur d'écriture, 
ni problème de majuscule. 


En résumé 


| 


Le code C# est composé d’une suite d’instructions qui se terminent par un point- 
virgule. 
La syntaxe d’un code C# doit être correcte sinon nous aurons des erreurs de com- 
pilation. 
Il est possible de commenter son code grâce aux caractères « //», « /*» et «*/». 
Visual C# Express dispose d’un outil puissant qui permet d’aider à compléter ses 
instructions : la complétion automatique. 
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Les variables 


Difficulté : 16: 


ans ce chapitre nous allons apprendre ce que sont les variables et comment ces 

éléments indispensables vont nous rendre bien des services pour traiter de l'informa- 

tion susceptible de changer dans nos programmes informatiques. Nous continuerons 
en découvrant les différents types de variables et nous ferons nos premières manipulations 
avec elles. 


Soyez attentifs à ce chapitre, il est vraiment fondamental de bien comprendre à quoi servent 
les variables lors du développement d'une application informatique. 
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Qu'est-ce qu’une variable ? 


Comme tous les langages de programmation, le C# va pouvoir conserver des données 
grâce à des variables. Ce sont en fait des blocs de mémoire qui vont contenir des 
nombres, des chaînes de caractères, des dates ou plein d’autres choses. Les variables 
vont nous permettre d'effectuer des calculs mathématiques, d'enregistrer par exemple 
pour un site, l’âge du visiteur, de comparer des valeurs, etc. On peut les comparer 
à des petits classeurs possédant une étiquette. On va pouvoir mettre des choses dans 
ces classeurs, par exemple, je mets 30 dans le classeur étiqueté « âge de Nicolas » et 
20 dans le classeur « âge de Jérémie ». Si je veux connaître l’âge de Nicolas, je n’ai 
qu’à regarder dans ce classeur pour obtenir 30. Je peux également remplacer ce qu’il 
y a dans mon classeur par autre chose, par exemple changer 30 en 25. Je ne peux par 
contre pas mettre deux choses dans mon classeur, il n’a qu’un seul emplacement. 


Une variable est représentée par son nom, caractérisée par son type et contient une 
valeur. 


caractères, une date, etc. 


Q Le type correspond à ce que la variable représente : un entier, une chaîne de 


Par exemple, l’âge d’une personne pourrait être stocké sous la forme d’un entier et 
accessible par la variable age, ce qui s'écrit en C=Æ : 


1 | int age; 


On appelle ceci « la déclaration de la variable age ». 


Le mot-clé int permet d'indiquer au compilateur que la variable age est un entier 
numérique. int correspond au début d’integer qui veut dire « entier » en anglais. 


Ici, la variable age n’a pas été initialisée, elle ne pourra pas être utilisée car le compi- 
lateur ne sait pas quelle valeur il y a dans la variable age. 


Pour l’initialiser (on parle également « d’affecter une valeur ») on utilisera l'opérateur 
égal («—»). 

1| int age; 

2| age = 30; 

Notre variable age possède désormais l’entier numérique « 30 » comme valeur. 


L’initialisation d’une variable peut également se faire au moment de sa déclaration. 
Aiünsi, on pourra remplacer le code précédent par : 


1 | int age = 30; 
Pour déclarer une variable en C}, on commence toujours par indiquer son type (int, 
ici un entier) et son nom (ici, age). Il faudra impérativement affecter une valeur à cette 


variable avec l’opérateur « — », soit sur la même instruction que la déclaration, soit un 
peu plus loin dans le code, mais dans tous les cas, avant l’utilisation de cette variable. 
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Nous pouvons à tout moment demander la valeur contenue dans la variable age, par 
exemple : 


1| int age = 30; 
2| Console.WriteLine(age); // affiche 30 


Il est possible de modifier la valeur de la variable à n'importe quel moment grâce à 
l'emploi de l’opérateur « — » que nous avons déjà vu : 


int age = 30; 

Console.WriteLine(age); // affiche 30 
age = 20; 

Console.WriteLine(age); // affiche 20 


& © ND 


Vous pouvez nommer vos variables à peu près n’importe comment, à quelques détails 
près. Les noms de variables ne peuvent pas avoir le même nom qu’un type. Il sera 
alors impossible d'appeler une variable int. Il est également impossible d’utiliser des 
caractères spéciaux, comme des espaces ou des caractères de ponctuation. De même, 
on ne pourra pas nommer une variable en commençant par des chiffres. 


Il est par contre possible d'utiliser des accents dans les noms de variables, 
cependant ceci n'est pas recommandé et ne fait pas partie des bonnes pra- 
tiques de développement. En effet, il est souvent recommandé de nommer ses 
variables en anglais (langue qui ne contient pas d'accents). Vous aurez noté 
que, volontairement, je ne le fais pas dans ce livre afin de ne pas ajouter une 
contrainte supplémentaire lors de la lecture du code. Mais libre à vous de le 
faire | 


En général, une variable commence par une minuscule et, si son nom représente plu- 
sieurs mots, on démarrera un nouveau mot par une majuscule. Par exemple : 


1 | int ageDuVisiteur; 
C’est ce qu’on appelle le camel case. 


Suivant le principe de sensibilité à la casse, il faut faire attention car 
ageduvisiteur et ageDuVisiteur seront deux variables différentes : 


int ageduvisiteur 30; 
int ageDuVisiteur 20; 
Console.WriteLine(ageduvisiteur); // affiche 30 
Console.WriteLine(ageDuVisiteur); // affiche 20 


& ND 
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À noter un détail qui peut paraître évident, mais toutes les variables sont 
réinitialisées à chaque nouvelle exécution du programme. Dès qu'on démarre 
le programme, les classeurs sont vidés, comme si on emménageait dans de 
nouveaux locaux à chaque fois. Il est donc impossible de faire persister une 
information entre deux exécutions du programme en utilisant des variables. 
Pour ceci, on utilisera d'autres solutions, comme enregistrer des valeurs dans 
un fichier ou dans une base de données. Nous y reviendrons ultérieurement. 


Les différents types de variables 


Nous avons vu juste au-dessus que la variable age pouvait être un entier numérique 
grâce au mot clé int. Le framework .NET dispose de beaucoup de types permettant 
de représenter beaucoup de choses différentes. 


Par exemple, nous pouvons stocker une chaîne de caractères grâce au type string. 


1 | string prenom = "nicolas"; 


ou encore un décimal avec : 


1 | decimal soldeCompteBancaire = 100; 


ou encore un boolean (qui représente une valeur vraie ou fausse) avec 


1 | bool estVrai = true; 


Il'est important de stocker des données dans des variables ayant le bon type. 


On ne peut par exemple pas stocker le prénom « Nicolas » dans un entier. 
Les principaux types de base du framework .NET sont : 


Vous verrez plus loin qu’il existe encore d’autres types dans le framework .NET et 
qu’on peut également construire les siens. 


Affectations, opérations, concaténation 


Il est possible d'effectuer des opérations sur les variables et entre les variables. Nous 
avons déjà vu comment affecter une valeur à une variable grâce à l’opérateur « = ». 


1|[ int age = 30; 
2| string prenom = "nicolas"; 
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Type Description 

byte Entier de 0 à 255 

short Entier de -32768 à 32767 

int Entier de -2147483648 à 2147483647 

long Entier de -9223372036854775808 à 9223372036854775807 

float Nombre simple précision de -3,402823e38 à 3,402823e38 

double Nombre double précision de -1,79769313486232e308 à 
1,79769313486232e308 

decimal Nombre décimal convenant particulièrement aux calculs financiers 
(en raison de ses nombres significatifs après la virgule) 

char Représente un caractère 

string Une chaîne de caractère 

bool Une valeur booléenne (vrai ou faux) 


Note : dans ce paragraphe, je vais vous donner plusieurs exemples d'affec- 
tations. Ces affectations seront faites sur la même instruction que la décla- 

Q ration pour des raisons de concision. Mais ces exemples sont évidemment 
fonctionnels pour des affectations qui se situent à un endroit différent de la 
déclaration. 


En plus de la simple affectation, nous pouvons également faire des opérations, par 
exemple : 


1 | int resultat = 2 * 3; 


ou encore 

1| int agei = 20; 

2| int age2 = 30; 

3| int moyenne = (agei + age2) / 2; 


Les opérateurs « +», « *»,« /» ou encore « -» (que nous n’avons pas encore utilisé) 
servent bien évidemment à faire les opérations mathématiques qui leur correspondent, 
à savoir respectivement l’addition, la multiplication, la division et la soustraction. Vous 
aurez donc sûrement deviné que la variable resultat contient 6 et que la moyenne vaut 


25. 


Il est à noter que les variables contiennent une valeur qui ne peut évoluer qu’en affectant 
une nouvelle valeur à cette variable. Ainsi, si j'ai le code suivant : 


1| int agei = 20; 

2| int age2 = 30; 

3| int moyenne = (agei + age2) / 2; 
4| age2 = 40; 


la variable moyenne vaudra toujours 25 même si j’ai changé la valeur de la variable 
age2. En effet, lors du calcul de la moyenne, j’ai rangé dans mon classeur la valeur 25 
grâce à l'opérateur d'affectation « — » et j'ai refermé mon classeur. Le fait de changer 
la valeur du classeur age2 n’influence en rien le classeur moyenne dans la mesure où il 
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est fermé. Pour le modifier, il faudrait réexécuter l’opération d'affectation de la variable 
moyenne, en écrivant à nouveau l'instruction de calcul, c’est-à-dire : 


1| int agei = 20; 

2| int age2 = 30; 

3| int moyenne = (agel + age2) / 2; 
4| age2 = 40; 

5 


moyenne = (agei + age2) / 2; 


L'opérateur + peut également servir à concaténer des chaînes de caractères, par exemple : 


1| string codePostal = "33000"; 

2| string ville = "Bordeaux"; 

3| string adresse = codePostal + " " + ville; 

4| Console.WriteLine (adresse); // affiche : 33000 Bordeaux 


D'autres opérateurs particuliers existent que nous ne trouvons pas dans les cours de 
mathématiques. Par exemple, l'opérateur ++ qui permet de réaliser une incrémentation 
de 1, ou l’opérateur -- qui permet de faire une décrémentation de 1. De même, les 
opérateurs que nous avons déjà vus peuvent se cumuler à l’opérateur — pour simplifier 
une opération qui prend une variable comme opérande et cette même variable comme 
résultat. Par exemple : 


age += 10; // équivalent à age = age + 10 (age contient 40) 
age /= 2; // équivalent à age = age / 2 => (age contient 20) 


1| int age = 20; 

2| age = age + 10; // age contient 30 (addition) 

3| age = age++; // age contient 31 (incrémentation de 1) 
al age = age--; // age contient 30 (décrémentation de 1) 
5 

6 


Comme nous avons pu le voir dans nos cours de mathématiques, il est possible de 
grouper des opérations avec des parenthèses pour agir sur leurs priorités. 


Ainsi, l'instruction précédemment vue : 


1 | int moyenne = (agel + age2) / 2; 


effectue bien la somme des deux âges avant de les diviser par 2, car les parenthèses sont 
prioritaires. 


Cependant, l'instruction suivante : 


1 | int moyenne = ageil + age2 / 2; 


aurait commencé par diviser age2 par 2 et aurait ajouté age1, ce qui n'aurait plus rien 
à voir avec une moyenne. En effet, la division est prioritaire par rapport à l’addition. 


Attention, la division ici est un peu particulière. 


Prenons cet exemple : 
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1 
2 


int moyenne = 5 / 2; 
Console.WriteLine (moyenne); 


Si nous l’exécutons, nous voyons que moyenne vaut 2. 
Q 2? Si je me rappelle bien de mes cours de maths, c'est plutôt 2.5, non ? 


Oui et non. Si nous divisions 5 par 2, nous obtenons bien 2.5. Par contre, ici nous 
divisons l’entier 5 par l’entier 2 et nous stockons le résultat dans l’entier moyenne. Le 
CF réalise en fait une division entière, c’est-à-dire qu'il prend la partie entière de 2.5, 
c’est-à-dire 2. 

De plus, l’entier moyenne est incapable de stocker une valeur contenant des chiffres 
après la virgule. Il ne prendrait que la partie entière. 


Pour avoir 2.5, il faudrait utiliser le code suivant : 


1 
2 


double moyenne = 5.0 / 2.0; 
Console.WriteLine (moyenne); 


Ici, nous divisons deux doubles entre eux et nous stockons le résultat dans un double. 
(Rappelez-vous, le type de données double permet de stocker des nombres à virgule.) 


Sans ça, il considère que les chiffres sont des entiers. 


d Le C# comprend qu'il s'agit de double car nous avons ajouté un .0 derrière. 


Les caractères spéciaux dans les chaînes de caractères 


En ce qui concerne l'affectation de chaînes de caractères, vous risquez d’avoir des sur- 
prises si vous tentez de mettre des caractères spéciaux dans des variables de type 
string. En effet, une chaîne de caractères étant délimitée par des guillemets «"!"», 
comment faire pour qu’elle puisse contenir des guillemets ? 


C’est là qu’intervient le caractère spécial \ qui sera à mettre juste devant le guillemet. 
Par exemple le code suivant : 


1] string phrase = "Mon prénom est \"Nicolas\""; 
2| Console.WriteLine (phrase); 
affichera : 


Mon prénom est "Nicolas" 


Si vous avez testé par vous-même l'instruction Console.WriteLine et enchaîné plu- 
sieurs instructions qui écrivent des lignes, vous avez pu remarquer que nous passions à 
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la ligne à chaque fois. C’est le rôle de l’instruction WriteLine qui affiche la chaîne de 
caractères et passe à la ligne à la fin de la chaîne de caractères. 


Nous pouvons faire la même chose en utilisant le caractère spécial « \n ». Il permet de 
passer à la ligne à chaque fois qu’il est rencontré. Ainsi, le code suivant : 


1| string phrase = "Mon prénom est \"Nicolas\""; 
2| Console.WriteLine (phrase); 
3| Console.WriteLine ("Passe\nä\nla\nligne\n\n\n"); 


affichera chaque mot précédé du caractère « \n » sur une ligne différente : 


Mon prénom est " Nicolas " 


Passe 


ligne 


Vous me diriez qu’on pourrait enchaîner les Console.WriteLine et vous auriez raison. 


Mais les caractères spéciaux nous permettent de faire d’autres choses comme une ta- 
bulation par exemple grâce au caractère spécial « \t ». Le code suivant : 


1| Console.WriteLine ("Choses à faire :"); 
2| Console.WriteLine("\t - Arroser les plantes"); 
3| Console.WriteLine("\t - Laver la voiture"); 


permettra d'afficher des tabulations : 


Choses à faire 


- Arroser les plantes 
- Laver la voiture 


Nous avons vu que le caractère \ était un caractère spécial et qu’il permettait de dire 
au compilateur que nous voulions l’utiliser combiné à la valeur qui le suit, pour avoir 
une tabulation où un retour à la ligne. Comment pourrons-nous avoir une chaîne de 
caractères qui contienne ce fameux caractère ? 


Le principe est le même, il suffira de faire suivre ce fameux caractère spécial de lui- 
même : 


1 
2 


string fichier = "c:\\repertoire\\fichier.cs"; 
Console.WriteLine(fichier); 


La console affichera alors les antislashs : 


c:\repertoire\fichier.cs 


Pour ce cas particulier, il est également possible d'utiliser la syntaxe suivante en utili- 
sant le caractère spécial @ devant la chaîne de caractères : 
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1| string fichier = @"c:\repertoire\fichier.cs"; // contient : c:\ 
repertoire\fichier.cs 


Bien sûr, nous pouvons stocker des caractères spéciaux dans des variables pour faire 
par exemple : 


1| string sautDeLigne = "\n'; 
2| Console.WriteLine("Passer" + sautDeLigne + "à" + 
3 sautDeLigne + "la" + sautDeLigne + "ligne"); 


Dans ce cas, la variable sautDeLigne peut être remplacée par une 
espèce de variable qui existe déjà dans le framework .NET, à savoir 
Environment .NewLine. 


Ce qui permet d’avoir le code suivant : 


1| Console.WriteLine("Passer" + Environment.NewLine + "à" + 
2| Environment .NewLine + "la" + Environment .NewLine + "ligne"); 


qui affichera : 


Notez qu'il est possible de passer à la ligne lors de l'écriture d'une instruction 
CH comme je l'ai fait dans le dernier bout de code afin d'améliorer la lisibilité. 
N'oubliez pas que c’est le point-virgule qui termine l'instruction. 


Environment .NewLine ? Une espèce de variable ? Qu'est-ce que c'est que 
cette chose-là ? 


En fait, je triche un peu sur les mots. Pour faciliter la compréhension, on peut considérer 
que Environment .NewLine est une variable, au même titre que la variable sautDeLigne 
que nous avons définie. En réalité, c’est un peu plus complexe qu’une variable. Nous 
découvrirons plus loin de quoi il s’agit vraiment. 


En résumé 


— Une variable est une zone mémoire permettant de stocker une valeur d’un type 
particulier. 

— Le C## possède plein de types prédéfinis, comme les entiers (int), les chaînes de 
caractères (string), etc. 


43 


CHAPITRE 4. LES VARIABLES 


— On utilise l'opérateur — pour affecter une valeur à une variable. 
— Il est possible de faire des opérations entre les variables. 
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Les instructions conditionnelles 


Difficulté : 1 


ans nos programmes C#, nous allons régulièrement avoir besoin de faire des opé- 

rations en fonction d'un résultat précédent. Par exemple, lors d'un processus de 

connexion à une application, si le login et le mot de passe sont bons, alors nous 
pouvons nous connecter, sinon nous afficherons une erreur. 


Il s'agit de ce que l'on appelle une condition. Elle est évaluée lors de l'exécution et en 
fonction de son résultat (vrai ou faux) nous ferons telle ou telle chose. 


Bien que relativement court, ce chapitre est très important. N'hésitez pas à le relire et à 
vous entraîner. 
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Les opérateurs de comparaison 


Une condition se construit grâce à des opérateurs de comparaison. On dénombre 
plusieurs opérateurs de comparaisons, les plus courants sont : 


Opérateur | Description 
—— Egalité 

= Différence 

> Supérieur à 

< Inférieur à 

>— Supérieur ou égal 
<= Inférieur ou égal 
&& ET logique 

|| OÙ logique 

! Négation 


Nous allons voir comment les utiliser en combinaison avec les instructions condition- 
nelles. 


L’instruction if 


L’instruction if permet d'exécuter du code si une condition est vraie !. Par exemple : 


1| decimal compteEnBanque = 300; 
2|[ if (compteEnBanque >= 0) 
3 Console.WriteLine ("Votre compte est créditeur"); 


Ici, nous avons une variable contenant le solde de notre compte en banque. Si notre 
solde est supérieur ou égal à 0 alors nous affichons que le compte est créditeur. 


Pour afficher que le compte est débiteur, on pourrait tester si la valeur de la variable 
est inférieure à 0 et afficher que le compte est débiteur : 


decimal compteEnBanque = 300; 
if (compteEnBanque >= 0) 

Console.WriteLine ("Votre compte est créditeur"); 
if (compteEnBanque < 0) 

Console.WriteLine ("Votre compte est débiteur"); 


œù & © ND 


Une autre solution est d'utiliser le mot-clé else, qui veut dire « sinon » en anglais. 


« Si la valeur est vraie, alors on fait quelque chose, sinon, on fait autre chose », ce qui 
se traduit en C# par : 


1| decimal compteEnBanque = 300; 
2|[ if (compteEnBanque >= 0) 
3 Console.WriteLine ("Votre compte est créditeur"); 


1. Le mot if signifie « si » en anglais. 
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4| else 
5 Console.WriteLine ("Votre compte est débiteur"); 


Il faut bien se rendre compte que l'instruction if teste si une valeur est vraie (dans 
l'exemple précédent la comparaison compteEnBanque >= 0). 


On a vu rapidement dans les chapitres précédents qu’il existait un type de variable 
qui permettait de stocker une valeur vraie ou fausse : le type bool, autrement appelé 
booléen (boolean en anglais). Ainsi, il sera également possible de tester la valeur d’un 
booléen. L'exemple précédent peut aussi s’écrire : 


decimal compteEnBanque = 300; 
bool estCrediteur = (compteEnBanque >= O0); 
if (estCrediteur) 
Console.WriteLine ("Votre compte est créditeur"); 
else 


Oo où À © ND 


Console.WriteLine ("Votre compte est débiteur"); 


Les parenthèses autour de l'instruction de comparaison sont facultatives, je les ai écrites 
ici pour clairement identifier que la variable estCrediteur va contenir une valeur qui 
est le résultat de l’opération de comparaison « compte en banque est supérieur ou égal 
à 0 », en l’occurrence vrai. Voici d’autres exemples pour vous permettre d'appréhender 
plus précisément le fonctionnement du type bool : 


int age = 30; 

bool estAgeDe30Ans = age == 30; 
Console.WriteLine(estAgeDe30Ans); // affiche True 
bool estSuperieurA10 = age > 10; 

Console.WriteLine (estSuperieurA10); // affiche True 
bool estDifferentDe30 = age != 30; 
Console.WriteLine(estDifferentDe30); // affiche False 


NO À & À D 


Un type bool peut prendre deux valeurs, vrai ou faux, qui s’écrivent avec les mots-clés 
true et false. 


1] bool estVrai = true; 

2| if (estVrai) 

3 Console.WriteLine("C'est vrai !"); 
4| else 

5 Console.WriteLine("C'est faux !"); 


Il est également possible de combiner les tests grâce aux opérateurs de logique condi- 
tionnelle, par exemple && qui correspond à l’opérateur « ET ». 


Dans l’exemple qui suit, nous affichons le message de bienvenue uniquement si le login 
est « Nicolas » ET que le mot de passe est « test ». Si l’un des deux ne correspond pas, 
nous irons dans l'instruction else. 


1] string login = "Nicolas"; 

2| string motDePasse = "test"; 

3| if (login == "Nicolas" && motDePasse == "test") 
4 Console.WriteLine("Bienvenue Nicolas"); 
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5| else 
6 Console.WriteLine("Login incorrect"); 


Remarquons ici que nous avons utilisé le test d'égalité ==, à ne pas confondre 
avec l'opérateur d'affection =. C'est une erreur classique de débutant! 


D’autres opérateurs de logique existent, nous avons notamment l'opérateur || qui cor- 
respond au « OÙ » logique : 


if (civilite == "Mme" || civilite == "Mlle'") 
Console.WriteLine("Vous êtes une femme"); 
else 


& © ND OH 


Console.WriteLine ("Vous êtes un homme"); 


L'exemple parle de lui-même ; si la civilité de la personne est Mme ou Mlle, alors nous 
avons à faire à une femme. 


Ici, si la première condition du if est vraie alors la deuxième ne sera pas évaluée. C’est 
un détail ici, mais cela peut s’avérer important dans certaines situations dont une que 
nous verrons un peu plus loin. 


Un autre opérateur très courant est la négation que l’on utilise avec l’opérateur «! ». 
Par exemple : 


1] bool estVrai = true; 

2| if (lestVrai) 

3 Console.WriteLine("C'est faux !"); 
al else 

5 Console.WriteLine("C'est vrai !"); 


Ce test pourrait se lire ainsi : si la négation de la variable estVrai est vraie, alors 
on écrira « C’est faux! » La variable « est Vrai » étant égale à true, sa négation vaut 
false. Dans cet exemple, le programme nous affichera donc l'instruction correspondant 
au else, à savoir « C’est vrai! » 


Rappelez-vous, nous avons dit qu’une instruction se finissait en général par un point- 
virgule. Comment cela se fait-il alors qu’il n’y ait pas de point-virgule à la fin du if 
ou du else ? Et si nous écrivions l’exemple précédent de cette façon ? 


1] bool estVrai = true; 
2| if (!estVrai) Console.WriteLine("C'est faux !"); 
3| else Console.WriteLine ("C'est vrai !"); 


Ceci est tout à fait valable et permet de voir où s'arrête vraiment l'instruction grâce 
au point-virgule. Cependant, nous écrivons en général ces instructions de la première 
façon afin que celles-ci soient plus lisibles. Vous aurez l’occasion de rencontrer dans les 
chapitres suivants d’autres instructions qui ne se terminent pas obligatoirement par un 
point-virgule. 
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Nous verrons dans le chapitre suivant, comment exécuter plusieurs instruc- 
tions après une instruction conditionnelle en les groupant dans des blocs de 
code, délimités par des accolades : { et }. 


Remarquons enfin qu’il est possible d’enchaîner les tests de manière à traiter plusieurs 
conditions en utilisant la combinaison else if. Cela donne : 


D I OO Où À À ND 


if (civilite == "Mme") 
Console.WriteLine("Vous êtes une femme"); 
else if (civilite == "Mlle") 
Console.WriteLine("Vous êtes une femme non mariée"); 
else if (civilite == "M.") 
Console.WriteLine("Vous êtes un homme"); 
else 
Console.WriteLine("Je n'ai pas pu déterminer votre civilité 


ni) 


L’instruction switch 


L’instruction switch peut être utilisée lorsqu'une variable prend beaucoup de valeurs. 
Elle permet de simplifier l'écriture. Ainsi, l'instruction suivante : 


NN OO 8 © NN 


String civilite = "M."; 

if (civilite == "M.") 
Console.WriteLine("Bonjour monsieur"); 

if (civilite == "Mme") 
Console.WriteLine("Bonjour madame"); 

if (civilite == "Mlle") 


Console.WriteLine("Bonjour mademoiselle"); 


pourra s’écrire : 


© ®Œ I OO OÙ À À ND 


EEE 
GO ND hr © 


String civilite = "M": 
switch (civilite) 
{ 
case "M." 
Console.WriteLine("Bonjour monsieur"); 
break; 
case "Mme": 
Console.WriteLine("Bonjour madame"); 
break; 
case "Mlle": 
Console.WriteLine("Bonjour mademoiselle"); 
break; 


} 


switch commence par évaluer la variable qui lui est passée entre parenthèses. Avec 
le mot-clé case on énumère les différents cas possibles pour la variable et on exécute 
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les instructions correspondantes jusqu’au mot-clé break qui signifie que l’on sort du 
switch. 


Nous pouvons également indiquer une valeur par défaut en utilisant le mot-clé default, 
ainsi dans l’exemple suivant tout ce qui n’est pas « M. », « Mme » ou « Mlle » donnera 
l’affichage d’un « Bonjour inconnu » : 


1| switch (civilite) 

2| { 

3 case CM," 

4 Console.WriteLine ("Bonjour monsieur" 
5 break; 

6 case "Mme": 

7 Console.WriteLine("Bonjour madame"); 
8 break; 

9 case "Mile": 


10 Console.WriteLine ("Bonjour mademoiselle"); 
11 break; 

12 default: 

13 Console.WriteLine ("Bonjour inconnu"); 

14 break; 

15] } 


Nous pouvons également enchaîner plusieurs cas pour qu’ils fassent la même chose, ce 
qui reproduit le fonctionnement de l'opérateur logique « OÙ » («|| »). Par exemple, 
on pourra remplacer l’exemple suivant : 


string mois = "Janvier"; 

if (mois == "Mars" || mois == "Avril" || mois == "Mai") 
Console.WriteLine ("C'est le printemps"); 

if (mois == "Juin" || mois == "Juillet" || mois == "Août") 
Console.WriteLine ("C'est l'été"); 

if (mois == "Septembre" || mois == "Octobre" || mois == " 
Novembre") 
Console.WriteLine ("C'est l'automne"); 


OO Où & À NN 


œ — 


if (mois == "Décembre" || mois == "Janvier" || mois == "Février 
LS) 


9 Console.WriteLine ("C'est l'hiver"); 


switch (mois) 

{ 
case "Mars": 
cèse “Avril”: 


ER 

2 

3 

4 

5 case "Mai": 
6 Console.WriteLine ("C'est le printemps"); 
7 break; 

8 case "juin: 

9 case "Juillet": 

10 case "Août": 


11 Console.WriteLine ("C'est l'été"); 
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12 
13 
14 
15 
16 
17 
18 
19 
20 
21 
22 
23 


} 


break; 

case "Septembre": 

case "Octobre": 

case "Novembre": 
Console.WriteLine ("C'est l'automne"); 
break; 

case "Décembre": 

case "Janvier": 

case "Février": 
Console.WriteLine ("C'est l'hiver"); 
break; 


Ce qui allège quand même l'écriture et la rend beaucoup plus lisible! 


En résumé 


— Les instructions conditionnelles permettent d'exécuter des instructions seulement si 


une condition est vérifiée. 


— On utilise en général le résultat d’une comparaison dans une instruction condition- 


nelle. 


— Le C# possède beaucoup d'opérateurs de comparaison, comme l’opérateur d'égalité 


—=—, l’opérateur de supériorité >, d’infériorité <, etc. 
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92 


Les blocs de code et la portée d'une 
variable 


Difficulté : 1 


N ous avons régulièrement utilisé dans le chapitre précédent les accolades ouvrantes et 


fermantes : { et }. Nous avons rapidement dit que ces accolades servaient à créer des 
blocs de code. 


L'utilisation d'accolades implique également une autre subtilité. Vous l'avez vu dans le titre 
du chapitre, il s'agit de la portée d'une variable. 


Regardons à présent comment cela fonctionne. 
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Les blocs de code 


Les blocs de code permettent de grouper plusieurs instructions qui vont s’exécuter 
dans le même contexte. Cela peut être le cas par exemple après un if, nous pourrions 
souhaiter effectuer plusieurs instructions. Par exemple : 


1| decimal compteEnBanque = 300; 

2|[ if (compteEnBanque >= 0) 

3| + 

4 Console.WriteLine ("Votre compte est créditeur"); 

5 Console.WriteLine("Voici comment ouvrir un livret ..."); 

6| } 

7| else 

8| + 

9 Console.WriteLine ("Votre compte est débiteur"); 

10 Console.WriteLine ("N'oubliez pas que les frais de dé 
Couvérté sont de ..;:"): 

ul} 


Ici, nous enchaînons deux Console .WriteLine en fonction du résultat de la comparai- 
son de compteEnBanque avec 0. 


Les blocs de code seront utiles dès qu’on voudra regrouper plusieurs instructions. C’est 
le cas pour les instructions conditionnelles mais nous verrons beaucoup d’autres utili- 
sations, comme le switch que nous avons vu juste avant, les boucles ou les méthodes 
que nous allons aborder dans le chapitre suivant. 


La portée d’une variable 


&« CE»... « portée »... on se croirait dans un cours de musique! 


En fait, la « portée d’une variable » est la zone de code dans laquelle une variable est 
utilisable. Elle correspond en général au bloc de code dans lequel est définie la variable. 


Ainsi, le code suivant : 


1| static void Main(string[] args) 

2| € 

3 string prenom = "Nicolas"; 

4 string civilite = "M."; 

5 if (prenom == "Nicolas") 

6 { 

7 int age = 30; 

8 Console.WriteLine("Votre âge est : " + age); 

9 switch (civilite) 

10 { 

11 case "M,": 

12 Console.WriteLine ("Vous êtes un homme de " + 
age + " ans"); 

13 break; 
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14 case "Mme: 

15 Console.WriteLine ("Vous êtes une femme de " + 
age + " ans"); 

16 break; 

17 } 

18 } 

19 if (age >= 18) 

20 { 

21 Console.WriteLine(prenom + ", vous êtes majeur"); 

22 } 

23| } 


est incorrect et provoquera une erreur de compilation. En effet, nous essayons d’accéder 
à la variable age en dehors du bloc de code où elle est définie. Nous voyons que cette 
variable est définie dans le bloc qui est exécuté lorsque le test d'égalité du prénom avec 
la chaîne « Nicolas » est vrai alors que nous essayons de la comparer à « 18 » dans un 
endroit où elle n’existe plus. 


Nous pouvons utiliser age dans tout le premier if, et dans les sous-blocs de code, 
comme c’est le cas dans le sous-bloc du switch, mais pas en dehors du bloc de code 
dans lequel la variable est définie. Ainsi, la variable prenom est accessible dans le dernier 
if car elle à été définie dans un bloc père. 


Vous noterez qu'ici, la complétion nous est utile. En effet, Visual C# Express propose 
de compléter le nom de la variable dans un bloc où elle est accessible. Dans un bloc où 
elle ne l’est pas, la complétion ne nous la propose pas. Comme Visual C# Express est 
malin, si la complétion ne propose pas ce que vous souhaitez, c'est que vous n’y avez 
pas le droit! Cela peut être parce que la portée ne vous l’autorise pas. Pour corriger 
l'exemple précédent, il faut déclarer la variable age au même niveau que la variable 
prenom. 


Q Ok, mais alors, pourquoi on ne déclarerait pas tout au début une bonne fois 


pour toutes ? Cela éviterait ces erreurs... non ? 


Eh non, ce n’est pas possible! Généralement, l’utilisation de variables accessibles de 
partout est une mauvaise pratique de développement. Même si on peut avoir un équi- 
valent en C7, il faut se rappeler que plus une variable est utilisée dans la plus petite 
portée possible, mieux elle sera utilisée et plus elle sera pertinente. Essayez donc de 
déterminer le bloc de code minimal où l’utilisation de la variable est adaptée. 


En résumé 
— Un bloc de code permet de regrouper des instructions qui commencent par { et qui 


finissent par }. 
— Une variable définie à l’intérieur d’un bloc de code aura pour portée ce bloc de code. 
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Les méthodes 


Difficulté : 16: 


lément indispensable de tout programme informatique, une méthode regroupe un 

ensemble d'instructions, pouvant prendre des paramètres et pouvant renvoyer une 

valeur. Lors de vos développements, vous allez avoir besoin de créer beaucoup de 
méthodes. 


Nous allons découvrir les méthodes dans ce chapitre mais nous y reviendrons petit à petit 
tout au long de l'ouvrage et vous aurez ainsi l'occasion d'approfondir vos connaissances. 


Vous pourrez trouver de temps en temps le mot « fonction » à la place du mot « mé- 
thode ». Cela signifie la même chose. C'est une relique du passé correspondant à un ancien 
mode de développement qui s'utilise de moins en moins, de même que le terme « procé- 
dure » qui est encore plus vieux! 
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Créer une méthode 


Le but de la méthode est de factoriser du code afin d'éviter d’avoir à répéter sans cesse 
le même code et ceci pour deux raisons essentielles : 


— déjà parce que l’homme est un être paresseux qui utilise son intelligence pour éviter 
tout travail inutile ; 

— ensuite parce que si jamais il y a quelque chose à corriger dans ce bout de code et 
s’il est dupliqué à plusieurs endroits, nous allons devoir faire une correction dans 
tous ces endroits. Si le code est factorisé à un unique endroit, nous ferons une unique 
correction. Oui, oui, encore la paresse mais cela permet aussi d'éviter d'oublier un 
bout de code dans un endroit caché. 


Ce souci de factorisation est connu comme le principe DRY, qui est l’acronyme des 
mots anglais Don't Repeat Yourself (Ne vous répétez pas). Le but est de ne jamais (à 
quelques exceptions près évidemment...) avoir à réécrire la même ligne de code. 


Par exemple, imaginons quelques instructions qui s'occupent d'écrire un message de 
bienvenue avec le nom de l’utilisateur. Le code C# pourrait être : 


1| Console.WriteLine ("Bonjour Nicolas"); 


2| Console.WriteLine("------- " + Environment .NewLine); 
3| Console.WriteLine("\tBienvenue dans le monde merveilleux du C#" 
J 


Dans l'instruction Console.WriteLine que nous utilisons régulièrement, 
WriteLine est une méthode. 


Si plus tard on veut réafficher le message de bienvenue, il faudra réécrire ces quatre 
lignes de code à moins que nous utilisions une méthode : 


1| static void AffichageBienvenue() 

21 { 

3 Console.WriteLine ("Bonjour Nicolas"); 

4 Console.WriteLine("------- " + Environment .NewLine); 

5 Console.WriteLine("\tBienvenue dans le monde merveilleux du 
CHR): 

6| } 


Dans l’exemple précédent, je définis une méthode qui s’appelle AffichageBienvenue. 


L’instruction : 


à| static void AffichageBienvenue() 
est ce qu’on appelle la signature de la méthode. Elle nous renseigne sur les paramètres 
de la méthode et sur ce qu’elle va renvoyer. Le mot-clé void signifie que la méthode 


ne renvoie rien. Les parenthèses vides à la fin de la signature indiquent que la méthode 
n’a pas de paramètre. 
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Q C'est la forme de la méthode la plus simple possible. 


Le mot-clé static ne nous intéresse pas pour l’instant, mais sachez qu’il sert à indiquer 
que la méthode est toujours disponible et prête à être utilisée. Dans ce contexte, il est 
obligatoire. Nous y reviendrons. 


En dessous de la signature de la méthode, nous retrouvons les accolades. Elles per- 
mettent de délimiter la méthode. Le bloc de code ainsi formé constitue ce qu’on appelle 
le « corps de la méthode ». 


En résumé, pour déclarer une méthode, nous aurons : 


1| Signature de la méthode 
2| € 
3 Corps de la méthode 
al} 


Nous pouvons désormais appeler cette méthode (c’est-à-dire l’exécuter), dans notre 
programme grâce à son nom. Par exemple, ici je l’appelle très facilement deux fois de 
suite : 


static void Main(string[] args) 
{ 

AffichageBienvenue(); 
AffichageBienvenue(); 


1 
2 
3 
4 
5| } 
6 
7| static void AffichageBienvenue() 
8 
9 


{ 
Console.WriteLine("Bonjour Nicolas"); 
10 Console.WriteLine("------- " + Environment .NewLine); 
11 Console.WriteLine("\tBienvenue dans le monde merveilleux du 
CRU); 
121} 


Et tout ça, sans effort ! C’est quand même plus simple et plus clair, non ? 


La méthode spéciale Main () 


La signature du premier bloc ci-dessus ne vous rappelle rien ? Mais si, elle ressemble 
beaucoup à celle de la méthode que nous avons créée précédemment. 


d Il s'agit d'une méthode spéciale, la méthode Main(). 


Elle à été générée par Visual C7 Express lorsque nous avons créé le projet Console. 


59 


CHAPITRE 7. LES MÉTHODES 


Cette méthode est en fait le point d’entrée de l’application, c’est-à-dire que quand le 
CLR tente d’exécuter notre application, il recherche cette méthode afin de pouvoir 
commencer à exécuter des instructions à partir d’elle. S'il ne la trouve pas, alors, il 
ne pourra pas exécuter notre application. C’est pour cela qu’il est important que cette 
méthode soit accessible de partout: rappelez-vous, c’est le rôle du mot-clé static 
que nous aurons l’occasion d’étudier plus en détail ultérieurement. Visual C7 Express 
nous garde bien de cette erreur. En effet, si vous supprimez cette méthode (ou que 
vous enlevez le mot-clé static) et que vous tentez de compiler notre application, vous 
aurez le message d'erreur suivant : 


Erreur 1! Le programme ?’C:\Users\Nico\Documents\Visual 
Studio 2010\Projects\C\#\MaPremiereApplication\ 


MaPremiereApplication\obj\x86\Release\MaPremiereApplication. 
exe?” ne contient pas une méthode ?’Main’ statique appropriée 
pour un point d’entrée 


Le message d’erreur est clair. Il a besoin d’une méthode Main() pour démarrer. 


Voilà pour cette méthode spéciale Main(). Elle est indispensable dans tout 
programme exécutable et c'est par-là que le programme démarre. 


Les lecteurs attentifs auront remarqué que cette méthode possède certains éléments 
dans la signature, entre les parenthèses... Des paramètres! Découvrons-les dans la 
prochaine section. .…. 


Paramètres d’une méthode 


Super, nous savons créer des méthodes. Nous allons pouvoir créer une méthode qui 
permet de souhaiter la bienvenue à la personne qui vient de se connecter à notre 
application. Si c’est Nicolas qui vient de se connecter, nous allons appeler la mé- 
thode AffichageBienvenueNicolas(). Si c’est Jérémie, nous appellerons la méthode 
AffichageBienvenueJeremie(), etc. 


1| static void AffichageBienvenueNicolas() 

21 { 

3 Console.WriteLine ("Bonjour Nicolas"); 

4 Console.WriteLine("------- " + Environment .NewLine); 

5 Console.WriteLine("\tBienvenue dans le monde merveilleux du 
CH"): 

6| } 

7 

8| static void AffichageBienvenueJeremie() 

9! € 

10 Console.WriteLine("Bonjour Jérémie"); 

11 Console.WriteLine("------- " + Environment .NewLine); 


PARAMÈTRES D'UNE MÉTHODE 


12 Console.WriteLine("\tBienvenue dans le monde merveilleux du 
CH); 
131 } 


Finalement, ce n’est pas si pratique que ça... Alors que nous venions juste d'évoquer 
le principe DRY, nous nous retrouvons avec deux méthodes quasiment identiques qui 
ne diffèrent que d’un tout petit détail. 


C’est là qu’interviennent les paramètres de méthode. Nous l’avons évoqué au para- 
graphe précédent, il est possible de passer des paramètres à une méthode. Ainsi, nous 
pourrons utiliser les valeurs de ces paramètres dans le corps de nos méthodes, les mé- 
thodes en deviendront d’autant plus génériques. 


Dans notre exemple d'affichage de message de bienvenue, il est évident que le nom de 
l’utilisateur sera un paramètre de la méthode. 


Les paramètres s’écrivent à l’intérieur des parenthèses qui suivent le nom de la méthode. 
Nous devons indiquer le type du paramètre ainsi que le nom de la variable qui le 
représentera au sein de la méthode. 


Il est possible de passer plusieurs paramètres à une méthode, on les séparera avec une 
virgule. Par exemple : 


1| static void DireBonjour (string prenom, int age) 

2| { 

3 Console.WriteLine("Bonjour " + prenom); 

4 Console.WriteLine("Vous avez " + age + " ans"); 
5| } 


Ici, la méthode DireBonjour prend en paramètres une chaîne de caractères prenom 
et un entier age. La méthode affiche « Bonjour » ainsi que le contenu de la variable 
prenom. De même, juste en dessous, elle affiche l’âge qui a été passé en paramètre. 


Nous pourrons appeler cette méthode de cette façon, depuis la méthode Main : 


1| static void Main(string[] args) 
2! { 
3 DireBonjour ("Nicolas", 30); 
4 DireBonjour ("Jérémie", 20); 
51 + 


Ce qui permettra d’afficher facilement un message de bienvenue personnalisé : 


Bonjour Nicolas 
Vous avez 30 ans 


Bonjour Jérémie 
Vous avez 20 ans 


Bien sûr, il est obligatoire de fournir en paramètre d’une méthode une variable de même 
type que le paramètre attendu. Dans le cas contraire, le compilateur est incapable de 
mettre la donnée qui à été passée dans le paramètre. D'ailleurs, si vous ne fournissez 
pas le bon paramètre, vous avez droit à une erreur de compilation. 
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Par exemple, si vous appelez la méthode avec les paramètres suivants : 


1| DireBonjour (10, 10); 
Vous aurez l'erreur de compilation suivante : 


impossible de convertir de ?’int” en string” 


Il est évidemment possible de passer des variables à une méthode, cela fonctionne de 
la même façon : 


1 
2 


string prenom = "Nicolas"; 
DireBonjour (prenom, 30); 


Nous allons revenir plus en détail sur ce qu’il se passe exactement lorsque nous abor- 
derons le chapitre sur le mode de passage des paramètres. 


Vous voyez, cela ressemble beaucoup à ce que nous avons déjà fait avec la méthode 
Console.WriteLine(). Facile, non ? 


La méthode Console .WriteLine fait partie de la bibliothèque du framework .NET. 
Elle est utilisée pour écrire des chaînes de caractères, des nombres ou plein d’autres 
choses sur la console. Le framework .NET contient énormément de méthodes utilitaires 
de toutes sortes. Nous y reviendrons un peu plus tard. 


Vous aurez peut-être remarqué un détail : nous avons préfixé toutes nos méthodes du 
mot-clé static. J’ai dit que c'était obligatoire dans notre contexte, pour être plus 
précis, c’est parce que la méthode Main() est statique que nous sommes obligés de 
créer des méthodes statiques. On a dit que la méthode Main() était obligatoirement 
statique parce qu’elle devait être accessible de partout afin que le CLR puisse trouver le 
point d’entrée de notre programme. Or, une méthode statique ne peut appeler que des 
méthodes statiques, c’est pour cela que nous sommes obligés (pour l'instant) de préfixer 
nos méthodes par le mot-clé static. Nous décrirons ce que recouvre exactement le 
mot-clé static dans la partie suivante. 


Retour d’une méthode 


Une méthode peut aussi renvoyer une valeur, par exemple un calcul. C’est souvent 
d’ailleurs son utilité première. 

On pourrait imaginer par exemple une méthode qui calcule la longueur de l’hypoténuse 
à partir des deux côtés d’un triangle rectangle. Sachant que a? + b? — c?, nous pouvons 
imaginer une méthode qui prenne en paramètres la longueur des deux côtés, fasse la 
somme de leurs carrés et renvoie la racine carrée du résultat. C’est ce que fait la 
méthode suivante : 


1| static double LongueurHypotenuse(double a, double b) 
2| € 

3 double sommeDesCarres = a * a + b * b; 

4 double resultat = Math.Sqrt (sommeDesCarres); 
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5 return resultat; 


6| } 


Vous aurez remarqué que la signature de la méthode commence par le mot-clé double, 
qui indique qu’elle va nous renvoyer une valeur de type double. Comme on l’a vu, 
double a et double b sont deux paramètres de la méthode et sont du type double. 


La méthode Math.Sgrt est une méthode du framework .NET, au même titre que la 
méthode Console.WriteLine, qui permet de renvoyer la racine carrée d’un nombre. 
Elle prend en paramètre un double et nous retourne une valeur de type double éga- 
lement qui correspond à la racine carrée du paramètre. C’est tout naturellement que 
nous stockons ce résultat dans une variable grâce à l’opérateur d'affectation =. 


À la fin de la méthode, le mot-clé return indique que la méthode renvoie la valeur à 
la méthode qui l’a appelée. Ici, nous renvoyons le résultat. 


Cette méthode pourra s’utiliser ainsi : 


1| static void Main(string[] args) 

2| € 

3 double valeur = LongueurHypotenuse(1, 3); 
4 Console.WriteLine (valeur); 

5 valeur = LongueurHypotenuse(10, 10); 

6 Console.WriteLine (valeur); 

7|} 


Comme précédemment, nous utilisons une variable pour stocker le résultat de l’exécu- 
tion de la méthode : 


3,16227766016838 


14,142135623731 


Notez qu'il est également possible de se passer d’une variable intermédiaire pour stocker 
le résultat. Ainsi, nous pourrons par exemple écrire : 


1| Console.WriteLine("Le résultat est : " + LongueurHypotenuse (1, 


3)); 


Avec cette écriture le résultat renvoyé par la méthode LongueurHypotenuse est direc- 
tement concaténé à la chaîne « Le résultat est : » et passé en paramètre à la méthode 
Console.WriteLine. 


Remarquez qu’on a fait l'opération : « a*a » pour mettre «a » au carré. On aurait 
également pu faire Math.Pow(a, 2) qui permet de faire la même chose à la différence 
que Pow permet de mettre à la puissance que l’on souhaite. Aïnsi, Math.Pow(a, 3) 
permet de mettre « a » au cube. 


Il faut savoir que le mot-clé return peut apparaître à n'importe quel endroit de la 
méthode. Il interrompt alors l’exécution de celle-ci et renvoie la valeur passée. Ce mot- 
clé est obligatoire, sans cela la méthode ne compile pas. Il est également primordial que 
tous les chemins possibles d’une méthode renvoient quelque chose. Les chemins sont 
déterminés par les instructions conditionnelles que nous avons vues précédemment. 
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Aiünsi, l'exemple suivant est correct : 


1| static string Conjugaison(string genre) 
2| € 

3 if (genre == "homme") 

4 return "é'"; 

5 else 

6 return "ée'; 

7| } 


car peu importe la valeur de la variable genre, la méthode renverra une chaîne. 


Alors que celui-ci : 


1| static string Conjugaison(string genre) 
2| € 

3 if (genre == "homme") 

4 return "é'"; 

5 else 

6 { 

7 if (genre == "femme") 

8 return "ée'; 

9 } 

10! } 


est incorrect. En effet, que renvoie la méthode si la variable genre contient autre chose 
que « homme » ou « femme » ? 


En général, Visual C# Express nous indiquera qu’il détecte un problème avec une 
erreur de compilation. 


Nous pourrons corriger ceci avec, par exemple : 


1| static string Conjugaison(string genre) 
2| { 

3 if (genre == "homme") 

4 return “el: 

5 else 

6 { 

7 if (genre == "femme") 
8 return "ée'; 

9 } 

10 return ""; 

11 } 


À noter que "" 


string.Empty. 


correspond à une chaîne vide et peut également s'écrire 


Nous avons vu dans le début du chapitre qu’il était possible de créer des méthodes qui 
ne retournent rien. Dans ce cas, on peut utiliser le mot-clé return sans valeur qui le 
suit pour stopper l’exécution de la méthode. Par exemple : 
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1| static void Bonjour(string prenom) 

2| { 

3 if (prenom == "inconnu") 

4 réturn; 

5 Console.WriteLine("Bonjour " + prenom); 
6| } 


Ainsi, si la variable prenom vaut « inconnu », alors nous quittons la méthode Bonjour 
et l’instruction Console .WriteLine ne sera pas exécutée. 


En résumé 


— Une méthode regroupe un ensemble d'instructions pouvant prendre des paramètres 
et pouvant renvoyer une valeur. 

— Les paramètres d’une méthode doivent être utilisés avec le bon type. 

— Une méthode qui ne renvoie rien est préfixée du mot-clé void. 

— Le point d’entrée d’un programme est la méthode statique Main(). 

— Le mot-clé return permet de renvoyer une valeur du type de retour de la méthode, 
à l’appelant de cette méthode. 
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Tableaux, listes et énumérations 


Difficulté : 1 


ans les chapitres précédents, nous avons pu utiliser les types de base du framework 
.NET, comme int, string, double, etc. Nous allons découvrir ici d'autres types qui 
vont s'avérer très utiles dans la construction de nos applications informatiques. 


Une fois bien maîtrisés, vous ne pourrez plus vous en passer ! 
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Les tableaux 


Voici le premier nouveau type que nous allons étudier, le type tableau. En déclarant 
une variable de type tableau, nous allons en fait utiliser une variable qui contient une 
suite de variables du même type. Prenons cet exemple : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 


Nous définissons ici un tableau de chaînes de caractères qui contient 7 chaînes de 
caractères : les jours de la semaine. Ne faites pas attention à l'opérateur new pour 
l’instant, nous y reviendrons plus tard: il permet simplement de créer le tableau. Les 
crochets « [| >» qui suivent le nom du type permettent de signaler au compilateur que 
nous souhaitons utiliser un tableau de ce type-là, ici le type string. 


Un tableau, c’est un peu comme une armoire dans laquelle on range des variables. 
Chaque variable est posée sur une étagère. Pour accéder à la variable qui est posée sur 
une étagère, on utilise le nom de l’armoire et on indique l’indice de l’étagère où est 
stockée la variable, en utilisant des crochets || : 


1| Console.WriteLine(jours[3]); // affiche Jeudi 

2| Console.WriteLine(jours[0]); // affiche Lundi 

3| Console.WriteLine(jours[10]); // provoque une erreur d'exé 
cution car l'indice n'existe pas 


Le premier élément du tableau se situe à l'indice O et le dernier se situe à 
l'indice « taille du tableau — 1 », c'est-à-dire 6 dans notre exemple. Si on 
tente d'accéder à un indice qui n'existe pas, l'application lèvera une erreur. 


Il est possible de parcourir l’ensemble d’un tableau avec l’instruction suivante : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| for (int i = 0; i < jours.Length; i++) 

31 € 

4 Console.WriteLine(jours[i]l); 

51} 


Nous parcourons ici les éléments de 0 à taille-1 et nous affichons l’élément du tableau 
correspondant à l’indice en cours. Ce qui nous donne : 


Lundi 
Mardi 
Mercredi 


Jeudi 
Vendredi 


Samedi 
Dimanche 


Revenons à présent sur la déclaration du tableau : 
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1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" ?}: 


Cette écriture permet de créer un tableau qui contient 7 éléments et d’affecter une 
valeur à chaque élément du tableau. Il s’agit en fait ici d’une écriture simplifiée. Cette 
écriture est équivalente à celle-ci : 


1| string[] jours = new string[7l]l; 
2| jours[0] = "Lundi"; 

3| jours[1] = "Mardi"; 

4a| jours[2] = "Mercredi"; 

5| jours[3] = "Jeudi"; 

6| jours[4] = "Vendredi"; 

7| jours[5] = "Samedi; 

8| jours[6] = "Dimanche"; 


ui est beaucoup plus verbeuse, mais d’un autre côté, plus explicite. 
? ? 


La première instruction crée un tableau qui peut contenir 7 éléments. C’est la taille 
du tableau, elle ne peut pas changer. Chaque instruction suivante affecte une valeur à 
un indice du tableau. Rappelez-vous, un tableau commence à l'indice 0 et va jusqu’à 
l'indice taille — 1. 


Il est possible facilement de faire des opérations sur un tableau, comme un tri. On 
pourra utiliser la méthode Array.Sort(). Par exemple : 


1 | Array.Sort(jours); 


Avec cette instruction, le tableau sera classé par ordre alphabétique. Vous aurez l’oc- 
casion de voir d’autres méthodes dans des chapitres ultérieurs. Aïnsi, le code suivant : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| Array.Sort(jours); 

3| for (int i = 0; i < jours.Length; i++) 

alt 

5 Console.WriteLine(jours[il); 

6| } 


produira un classement alphabétique des entrées du tableau : 


Dimanche 
Jeudi 


Lundi 


Mardi 
Mercredi 
Samedi 
Vendredi 


Ce qui est très inutile! 


Le tableau jours est ce que l’on appelle un tableau à une dimension. Il est également 
possible de créer des tableaux à N dimensions, il est cependant assez rare de dépasser 
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deux dimensions. C’est surtout utile lorsqu'on manipule des matrices. 


Nous n’étudierons pas les tableaux à plus d’une dimension dans ce livre car ils risquent 
de vraiment peu vous servir dans vos premières applications. Par contre, le type «liste », 
que nous allons voir tout de suite, vous servira abondamment ! 


Les listes 


Un autre type que nous allons beaucoup utiliser est la liste. Nous allons voir comment 
ce type fonctionne mais sans en faire une étude exhaustive car elle pourrait être bien 
longue et ennuyeuse. Regardons cet exemple : 


1| List<int> chiffres = new List<int>(); // création de la liste 
2| chiffres.Add(8); // chiffres contient 8 

3| chiffres.Add(9); // chiffres contient 8, 9 

4| chiffres.Add(4); // chiffres contient 8, 9, 4 

5 

6| chiffres.RemoveAt (1); // chiffres contient 8, 4 
7 

8| foreach (int chiffre in chiffres) 

CEE 

10 Console.WriteLine(chiffre); 

1%] 


La première ligne permet de créer la liste. Nous reviendrons sur cette instruction un 
peu plus loin dans ce chapitre. Il s’agit d’une liste d’entiers. 


Nous ajoutons des entiers à la liste grâce à la méthode Add(). Nous ajoutons en l’oc- 
currence les entiers 8, 9 et 4. 


La méthode RemoveAt() permet de supprimer un élément en utilisant son indice, ici 
nous supprimons le deuxième entier, c’est-à-dire 9. 


Comme les tableaux, le premier élément de la liste commence à l'indice 0. 


Après cette instruction, la liste contient les entiers 8 et 4. 


Enfin, nous parcourons les éléments de la liste grâce à l’instruction foreach. Nous 
y reviendrons en détail lors du chapitre sur les boucles. Pour l'instant, nous avons 
juste besoin de comprendre que nous affichons tous les éléments de la liste, comme 
ci-dessous : 


Les lecteurs assidus auront remarqué que la construction de la liste est un peu particu- 
lière. On observe en effet l’utilisation de chevrons <> pour indiquer le type de la liste. 
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Pour avoir une liste d’entier, il suffit d’indiquer le type int à l’intérieur des chevrons. 
Ainsi, il ne sera pas possible d'ajouter autre chose qu’un entier dans cette liste. Par 
exemple, l’instruction suivante provoque une erreur de compilation : 


1| List<int> chiffres = new List<int>(); // création de la liste 
2| chiffres.Add("chaîne"); // ne compile pas 


Évidemment, on ne peut pas ajouter des choux à une liste de carottes ! 


De la même façon, si l’on souhaite créer une liste de chaîne de caractères, nous pourrons 
utiliser le type string à l’intérieur des chevrons : 


1| List<string> chaînes = new List<string>(); // création de la 
liste 

2| chaînes.Add("chaîne"); // compilation OK 

3| chaînes.Add(1); // compilation KO, on ne peut pas ajouter un 
entier dans une liste de chaînes de caractères 


Les listes possèdent des méthodes bien pratiques qui permettent toutes sortes d’opé- 
rations. Par exemple, la méthode IndexOf () permet de rechercher un élément dans la 
liste et de renvoyer son indice. 


List<string> jours = new List<string>(); 
jours.Add('"Lundi"); 

jours.Add("Mardi"); 
jours.Add("Mercredi"); 
jours.Add("Jeudi"); 
jours.Add("Vendredi"); 
jours.Add("Samedi"); 
jours.Add("Dimanche"); 


© © I OO Où À CO ND 


H 
=) 


int indice = jours.IndexOf ("Mercredi"); // indice vaut 2 


Nous aurons l’occasion de voir d’autres utilisations de méthodes de la liste dans les 
chapitres suivants. 


La liste que nous venons de voir (List<>) est en fait ce que l’on appelle un type 
générique. Nous n’allons pas rentrer dans le détail de ce qu’est un type générique 
pour l'instant, mais il faut juste savoir qu’un type générique permet d’être spécialisé 
par un type concret. Pour notre liste, cette généricité permet d'indiquer de quel type 
est la liste, une liste d’entiers ou une liste de chaînes de caractères, etc. 


Ne vous inquiétez pas si tout ceci n’est pas parfaitement clair, nous reviendrons plus 
en détail sur les génériques dans un chapitre ultérieur. Le but ici est de commencer à 
se familiariser avec le type List<> que nous utiliserons régulièrement et les exemples 
que nous verrons permettront d'appréhender les subtilités de ce type. 


À noter qu'il existe également une écriture simplifiée des listes. En effet, il est possible 
de remplacer : 


1| List<string> jours = new List<string>(); 
2| jours.Add("Lundi"); 
3| jours.Add("Mardi"); 
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4| jours.Add("Mercredi"); 

5| jours.Add("Jeudi"); 

6| jours.Add("Vendredi"); 

7| jours.Add("Samedi"); 

8| jours.Add("Dimanche"); 

par 

1| List<string> jours = new List<string> { "Lundi", "Mardi", " 


Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" }; 


un peu comme pour les tableaux. 


Liste ou tableau ? 


Vous aurez remarqué que les deux types, tableau et liste, se ressemblent beaucoup. 
Essayons de voir ce qui les différencie afin de savoir quel type choisir entre les deux. 


En fait, une des grosses différences est que le tableau peut être multidimensionnel. C’est 
un point important mais dans le cadre de vos premières applications, il est relativement 
rare d’avoir à s’en servir. Cela sert le plus souvent dans le développement de jeux ou 
lorsque l’on souhaite faire des calculs 3D. 


Par contre, le plus important pour nous est que le type tableau est de taille fixe alors 
que la liste est de taille variable. On peut ajouter sans problème un nouvel élément à 
une liste grâce à la méthode Add(). De même, on peut supprimer des éléments avec 
la méthode Remove alors qu'avec un tableau, on peut seulement remplacer les valeurs 
existantes et il n’est pas possible d'augmenter sa capacité. 


Globalement, vous verrez que dans vos applications, vous utiliserez plutôt les listes, 
par exemple pour afficher une liste de produits, une liste de clients, etc. 


Gardez quand même dans un coin de l'esprit les tableaux, ils pourront vous aider dans 
des situations précises. 


Les énumérations 


Un type particulier que nous allons également utiliser est l’énumération. Cela corres- 
pond comme son nom l'indique à une énumération de valeurs. 


Par exemple, il pourrait être très facile de représenter les jours de la semaine dans une 
énumération plutôt que dans un tableau. 


On définit l’énumération de cette façon, grâce au mot-clé enun : 


1| enum Jours 

2| À 

3 Lundi, 

4 Mardi, 

5 Mercredi, 
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6 Jeudi, 

7 Vendredi, 
8 Samedi, 

9 Dimanche 
10! } 


À noter qu’on ne peut pas définir cette énumération n'importe où, pour l'instant, 
contentons-nous de la définir en dehors de la méthode Main(). 


Pour être tout à fait précis, une énumération est un type dont toutes les valeurs dé- 
finies sont des entiers. La première vaut 0, et chaque valeur suivante prend la valeur 
précédente augmentée de 1. C'est-à-dire que Lundi vaut 0, Mardi vaut 1, etc. 


Il est possible de forcer des valeurs à toutes ou certaines valeurs de l’énumération, les 
valeurs non forcées prendront la valeur précédente augmentée de 1 : 


1| enum Jours 

2| À 

3 Lundi = 5, // lundi vaut 5 

4 Mardi, // mardi vaut 6 

5 Mercredi = 9, // mercredi vaut 9 
6 Jeudi = 10, // jeudi vaut 10 

7 Vendredi, // vendredi vaut 11 

8 Samedi, // samedi vaut 12 

9 Dimanche = 20 // dimanche vaut 20 
10! } 


Mais, à part pour enregistrer une valeur dans une base de données, il est rare de 
manipuler les énumérations comme des entiers car le but de l’énumération est justement 
d’avoir une liste exhaustive et fixée de valeurs constantes. Le code s’en trouve plus clair, 
plus simple et plus lisible. 


Le fait de définir une telle énumération revient en fait à enrichir les types que nous avons 
à notre disposition, comme les entiers ou les chaînes de caractères (int ou string). Ce 
nouveau type s'appelle Jours. 


Nous pourrons définir une variable du type Jours de la même façon qu'avec un autre 
type : 

1 | Jours jourDeLaSemaine; 

La seule différence c’est que les valeurs qu’il est possible d’affecter à notre variable sont 
figées et font partie des valeurs définies dans l’énumération. 


Pour pouvoir accéder à un élément de l’énumération, il faudra utiliser le nom de l’énu- 
mération suivi de l’opérateur point «.», suivi encore de la valeur de l’énumération 
choisie. Par exemple : 


1| Jours lundi = Jours.Lundi; 
2| Console.WriteLine (lundi); 


soit dans notre programme : 


1 | class Program 
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2! À 

3 enum Jours 

4 { 

b Lundi, 

6 Mardi, 

7 Mercredi, 

8 Jeudi, 

9 Vendredi, 

10 Samedi, 

11 Dimanche 

12 } 

13 

14 static void Main(string[] args) 
15 { 

16 Jours lundi = Jours.Lundi; 
17 Console.WriteLine (lundi); 
18 } 

19 | } 


La console nous affichera : 


Nous pourrons également nous servir des énumérations pour faire des tests de compa- 
raison, comme par exemple : 


1| if (jourDeLaSemaine == Jours.Dimanche || jourDeLaSemaine == 
Jours.Samedi) 
{ 
3 Console.WriteLine('"Bon week-end"); 
al} 


Sachez que le framework .NET utilise beaucoup les énumérations. Il est important de 
savoir les manipuler. 


Vous aurez peut-être remarqué que lorsqu'on affiche la valeur d'une énumé- 
ration, la console nous affiche le nom de l'énumération et non pas sa valeur 
entière. Ça peut paraître étrange, mais c'est parce que le C# fait un traite- 
ment particulier dans le cadre d'une énumération à l'affichage. 


En résumé 


— Un tableau est un type évolué pouvant contenir une séquence d’autres types, comme 
un tableau d’entiers ou un tableau de chaînes de caractères. 

— Une liste est un type complexe un peu plus souple que le tableau permettant d’avoir 
une liste de n’importe quel type. 

— Une énumération s'utilise lorsque l’on veut créer un type possédant plusieurs valeurs 
fixes, comme les jours de la semaine. 
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Difficulté : 1 
omme on l'a déjà évoqué, le framework .NET est une énorme boîte à outils qui 
contient beaucoup de méthodes permettant de construire toutes sortes d'applications. 


Nous allons avoir besoin régulièrement d'utiliser les éléments du framework .NET pour 
réaliser nos applications. Il est donc grand temps d'apprendre à savoir le manipuler! 
Rentrons tout de suite dans le vif du sujet. 
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L’instruction using 


Nous allons sans arrêt solliciter la puissance du framework .NET. Par exemple, nous 
pouvons lui demander de nous donner la date courante. 


Pour ce faire, on utilisera l’instruction : 

1 | Console.WriteLine (DateTime.Now); 

Ce qui se passe ici, c’est que nous demandons à notre application l’affichage de la 
propriété Now de l’objet DateTime. Nous allons revenir en détail sur ce que sont des 


propriétés et des objets, considérez pour l’instant qu’ils correspondent simplement à 
une instruction qui nous fournit la date du moment : 


09/08/2011 23:47:58 


En fait, pour accéder à la date courante, on devrait normalement écrire : 


1 | System.Console.WriteLine (System.DateTime.Now); 


En effet, les objets DateTime et Console se situent dans l’espace de nom System. 


Un espace de nom | correspond à un endroit où l’on range des méthodes et des objets. 
Il est caractérisé par des mots séparés par des points « .». 


C’est un peu comme des répertoires, nous pouvons dire que le fichier DateTime est 
rangé dans le répertoire « System » et, quand nous souhaitons y accéder, nous devons 
fournir l’emplacement complet du fichier, à savoir System.DateTime. 


Cependant, plutôt que d'écrire le chemin complet à chaque fois, il est possible de dire : 
«OK, maintenant, à chaque fois que je vais avoir besoin d’accéder à une fonctionnalité, 
va la chercher dans l’espace de nom System. Si elle s’y trouve, utilise-la ». 


C’est ce qui se passe grâce à l’instruction suivante : 


1 | using System; 


qui a été générée par Visual C# Express au début du fichier. 


Elle indique au compilateur que nous allons utiliser l’espace de nom System dans notre 
page. Ainsi, il n’est plus obligatoire de préfixer l’accès à DateTime par System. C’est 
une espèce de raccourci. Pour conserver l’analogie avec les répertoires et les fichiers, on 
peut dire que nous avons ajouté le répertoire « System » dans le path. 


Pour résumer, l'instruction : 


1 | System.Console.WriteLine (System.DateTime.Now); 


est équivalente aux deux instructions : 


1 | using System; 


et 


1. En anglais, namespace 
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1] Console.WriteLine (DateTime.Now) ; 


Sachant qu’elles ne s’écrivent pas côte à côte. En général, on met l'instruction using 
en en-tête du fichier .cs, comme ce qu'a fait Visual C# Express lorsqu'il a généré le 
fichier. L'autre instruction est à positionner à l’endroit où nous souhaitons qu’elle soit 
exécutée. 


Express ne vous proposera pas le mot DateTime. C'est un bon moyen de se 
rendre compte qu'il manque la déclaration de l’utilisation de l'espace de nom. 


@ Si le using System est absent, la complétion automatique de Visual C# 


À noter que dans ce cas-là, si Visual C# Express reconnaît l'instruction mais que 
l’espace de nom n’est pas inclus, il le propose en soulignant le début du mot DateTime 
d’un petit trait bleu et blanc, comme l’illustre la figure 9.1. 


class Program 
{ 


static void Main(string[] args) 


DateTime 
} 
7 
FIGURE 9.1 — Le trait bleu permet d'indiquer que l’'IDE sait résoudre l'import de 
l’espace de nom 


Un clic droit sur le mot permettra d'ouvrir un menu déroulant, de choisir Résoudre et 
d'importer le using correspondant automatiquement (voir figure 9.2). 


Un void Main(string[] args) 


DateT/: 
ee some 
Refactoriser 4 System.DateTime 


Organiser les instructions Using > 
ER Insérer un extrait. Ctri+K, X 


FIGURE 9.2 — Importer l’espace de nom automatiquement 


La bibliothèque de classes .NET 


Vous aurez l’occasion de découvrir que le framework .NET fourmille d'espaces de noms 
contenant plein de méthodes de toutes sortes permettant de faire un peu tout et n’im- 
porte quoi. Une vraie caverne d’Ali Baba! 


Parmi les nombreuses fonctionnalités du framework .NET, nous avons une méthode 
qui nous permet de récupérer l'utilisateur courant (au sens « utilisateur Windows »), 
on pourra par exemple utiliser l’instruction suivante : 
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1 | Console.WriteLine (System.Environment .UserName); 


qui affichera l'utilisateur Windows dans la console : 


Ou, comme vous le savez désormais, on pourra utiliser conjointement : 


1 | using System; 


et 


1 | Console.WriteLine (Environment .UserName) ; 


pour produire le même résultat. 


Petit à petit vous allez retenir beaucoup d’instructions et d’espaces de nom du fra- 
mework .NET mais il est inimaginable de tout retenir. Il existe une documentation 
répertoriant tout ce qui compose le framework .NET, c’est ce qu'on appelle la MSDN 
library. Vous pouvez y accéder via le code web suivant : 


MSDN library 
Code web : 153597 


Par exemple, la documentation de la propriété que nous venons d’utiliser est consultable 
avec le code web suivant : 


Doc UserName 
Code web : 563051 


Nous avons désigné la caverne d’Ali Baba par le mot « framework .NET ». Pour être 
plus précis, la désignation exacte de cette caverne est : «la bibliothèque de classes 
.NET ». Le mot « framework .NET » est en général utilisé par abus de langage (et 
vous avez remarqué que j'ai moi-même abusé du langage!) et représente également 
cette bibliothèque de classes. Nous reviendrons plus tard sur ce qu’est exactement une 
classe, pour l'instant, vous pouvez voir ça comme des composants. Le framework serait 
donc une bibliothèque de composants. 


Référencer une assembly 


Ca y est, nous savons accéder au framework .NET. 


du framework .NET sans nous poser de question ? Il est magique ce frame- 
work ? Où se trouve le code qui permet de récupérer la date du jour ? 


Q Mais d'ailleurs, comment se fait-il que nous puissions accéder aux méthodes 


Judicieuse question. Si on y réfléchit, il doit falloir beaucoup de code pour renvoyer la 
date du jour ! Déjà, il faut aller la lire dans l’horloge système, il faut peut-être l’adapter 
au fuseau horaire, la formater d’une façon particulière en fonction de la langue, etc. 
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Tout ça est déjà fait, heureusement, dans la bibliothèque de méthodes du framework 
NET. 


© Alors, où est-elle cette bibliothèque ? Et le reste ? 


Dans des assemblys bien sûr ! Comme on l’a vu, les assemblys possèdent des fragments 
de code compilés en langage intermédiaire. S’ils sont réutilisables, ils se trouvent dans 
des fichiers dont l’extension est .d11. 


Le framework .NET est composé d’une multitude d’assemblys qui sont installées sur 
votre système d'exploitation, dans le « GAC » ? qui est un endroit où sont stockées ces 
assemblys afin de les rendre accessibles depuis nos programmes. Pour pouvoir utiliser 
ces assemblys dans notre programme, nous devons indiquer que nous voulons les utiliser. 
Pour ça, il faut les référencer dans le projet. Si l’on déplie les références dans notre 
explorateur de solutions, nous pouvons voir que nous avons déjà pas mal de références 
qui ont été ajoutées par Visual C# Express (voir figure 9.3). 


[4 Solution 'MaPremiereApplication' (1 projet) | 
4 (] MaPremiereApplication 
b [54] Properties 
4 Références 
- Microsoft.CSharp 
- System 


3 System.Core 

-Q System.Data 

-& System.Data.DataSetExtensions 
-Q System.Xml 

© System.Xml.Linq 

Program.cs 


FIGURE 9.3 — Les références de notre application console 


Ce sont des assemblys qui sont très souvent utilisées, c’est pour ça que Visual C# 
Express nous les a automatiquement référencées. Toujours ce souci de nous simplifier 
le travail. Qu'il est sympa! 


Prenons par exemple la référence System.Xml. Son nom nous suggère que dedans 
est réuni tout ce qu'il nous faut pour manipuler le XML. Commençons par taper 
System.Xml., la complétion automatique nous propose plusieurs choses (voir figure 
9.4). 


On ne sait pas du tout à quoi elle sert, mais déclarons par exemple une variable de 
l’énumération ConformanceLevel : 


2. GAC est l’acronyme de Global Assembly Cache. 
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static void Main(string[] args) 


{ 
System.Xml. 
} &P ConformanceLevel + | enum System.Xml.ConformanceLevel 
&F DtdProcessing (E) Spécifie l'ampleur des contrôles d'entrée ou de sortie que les 
 EntityHandling System.Xml.XmlWriter créés effectuent. 
sf Formatting 


9 IHasXmINode 

2 IXmilLinelnfo 

2 IXmINamespaceResolver 

{} Linq 

SF NamespaceHandling M 


FIGURE 9.4 — L'espace de nom System.Xml est accessible si la référence est présente 


1 | System.Xml.ConformanceLevel level; 


et compilons. Pas d’erreur de compilation. Supprimez la référence à System.Xm1l. (clic 
droit, Supprimer) et compilez à nouveau, vous verrez que ConformanceLevel est dé- 
sormais souligné en rouge, signe qu’il y à un problème (voir figure 9.5). 


9 IVIGF TETTNET EAP PNCGLIUTLE FUYIATTI 737 MAMNSLTINUYL] arY>) 
E class Program 
L 
E static void Main(string[] args) 
{ 
System.Xml.ConformanceLevel level; 
} 
| } 
100% + 4 
Liste d'erreurs 
|@ 1 erreur | AA 0 avertissements || {i) 0 messages | 
Description | 


@1 Letype ou le nom d'espace de noms ‘'ConformanceLevel' n'existe pas dans l'espace de noms 'System.Xml' (une 
référence d'assembly est-elle manquante ?) 


FIGURE 9.5 — Erreur de compilation car la référence a été supprimée 


Par ailleurs, la compilation provoque l'erreur suivante : 
; P P q 


Le type ou le nom d’espace de noms ?’ConformanceLevel? n’existe 


pas dans l’espace de noms ?’System.Xml’ (une référence d’? 
assembly est-elle manquante ?) 


Loin d’être bête, Visual C# Express nous affiche un message d’erreur plutôt explicite. 


En effet, cette énumération est introuvable car elle est définie dans une assembly qui 
n’est pas référencée. C’est comme si on nous demandait de prendre le marteau pour 
enfoncer un clou, mais que le marteau n’est pas à côté de nous, mais au garage bien 
rangé ! Il faut donc référencer le marteau afin de pouvoir l'utiliser. Pour ce faire, faisons 
un clic sur Références et ajoutons une référence (voir figure 9.6). 


Ici, nous avons plusieurs onglets (voir figure 9.7). Selon la version de Visual Studio que 
vous possédez, vous aurez peut-être une présentation légèrement différente. 
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Cd Solution 'MaPremiereApplication' (1 projet) 
4 (1 MaPremiereApplication 
(54 Properties 


FIGURE 9.6 — Ajouter une référence 


Filtré pour : .NET Framework 4 Client Profile 
Nom du composant Version Runtime Chemin d'acc * 
Accessibility 4.0.0.0 v4030319 C:\Program FL 
CustomMarshalers 40.00 v4.0,30319 C:\Program F 
Microsoft.CSharp 4.0.0.0 v40.30319  C:\ProgramF 
Microsoft.JScript 10.0.0.0 v4.0.30319 C:\Program F 
Microsoft.VisualBasic.Compatibility.Data 10.0.0.0 v40.30319  C:\ProgramF 
Microsoft.VisualBasic.Compatibility 10.0.0.0 v4.0.30319 C:\Program F 


Microsoft.VisualBasic 10.0.0.0 v40.30319  C:\ProgramF 
Microsoft. VisualC 10.0.0.0 v4.0.30319 C:\Program F 
mscorlib 40.00 v40.30319  C:\ProgramF 
PresentationCore 40.00 v4.0.30319 C:\Program F 
PresentationFramework.Aero 40.00 v4030319  C:\ProgramF 
PresentationFramework.Classic 4.0.0.0 v40.30319  C:\ProgramF | 


«[ 


FIGURE 9.7 — Liste des assemblys présentes dans le GAC pouvant être référencées 
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Chaque onglet permet d’ajouter une référence à une assembly. 


— L’onglet .NET permet de référencer une assembly présente dans le GAC, c’est ici que 
nous viendrons chercher les assemblys du framework .NET. 

— L’onglet COM permet de référencer une dll COM. Vous ne savez pas ce qu'est COM ? 
On peut dire que c’est l’ancêtre de l’assembly. Techniquement, on ne référence pas di- 
rectement une dll COM, mais Visual C# Express génère ce qu’on appelle un wrapper 
permettant d'utiliser la dl COM. Un wrapper est en fait une espèce de traducteur 
permettant d'utiliser la dl COM comme si c'était une assembly. Mais ne nous attar- 
dons pas sur ce point qui ne va pas nous servir, nous qui sommes débutants. 

— L'onglet Projets permet de référencer des assemblys qui se trouvent dans notre 
solution. Dans la mesure où notre solution ne contient qu’un seul projet, l'onglet sera 
vide. Nous verrons plus tard comment utiliser cet onglet lorsque nous ajouterons des 
assemblys à nos solutions. 

— L’onglet Parcourir va permettre de référencer une assembly depuis un emplacement 
sur le disque dur. Cela peut-être une assembly tierce fournie par un collègue, ou par 
un revendeur, etc. 

— Enfin, l’onglet Récent, comme son nom le suggère, permet de référencer des assem- 
blys récemment référencées. 


Retournons à nos moutons, repartons sur l’onglet .NET et recherchons l’assembly que 
nous avons supprimée, à savoir System.Xml. 


Une fois trouvée, appuyez sur OK. Maintenant que la référence est ajoutée, nous pouvons 
à nouveau utiliser cette énumération... Ouf! 


Je ne comprends pas, j'ai supprimé toutes les références, mais j'arrive 
quand même à utiliser la date, le nom de l'utilisateur ou la méthode 
Console.WriteLine. Si j'ai bien compris, je ne devrais pas pouvoir utiliser 


: des méthodes dont les assemblys ne sont pas référencées... C'est normal ? 


Eh oui, il existe une assembly qui n'apparaît pas dans les références et qui contient 
tout le cœur du framework .NET. Cette assembly doit obligatoirement être référencée, 
il s’agit de mscorlib.d11. Comme elle est obligatoire, elle est référencée par défaut dès 
que l’on crée un projet. 


D’autres exemples 


Il arrivera souvent que vous ayez besoin d’une fonctionnalité trouvée dans la docu- 
mentation ou sur internet et qu'il faille ajouter une référence. Ce sera peut-être une 
référence au framework .NET, à des bibliothèques tierces ou à vous. Nous verrons plus 
loin comment créer nos propres bibliothèques. 


Dans la documentation MSDN, il est toujours indiqué quelle assembly il faut référencer 
pour utiliser une fonctionnalité. Prenons par exemple la propriété DateTime.Now, la 
documentation nous dit : 
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Espace de noms : System 


Assembly : mscorlib (dans mscorlib.d1ll) 


Cela veut donc dire qu’on utilise la date du jour en utilisant l’assembly obligatoire 
mscorlib et la fonctionnalité se trouve dans l’espace de nom System, comme déjà vu. 


I n’y a donc rien à faire pour pouvoir utiliser DateTime.Now. En imaginant que nous 
ayons besoin de faire un programme qui utilise des nombres complexes, vous allez 
sûrement avoir besoin du type Complex, trouvé dans la documentation adéquate, via 
le code web suivant : 


Doc complex 
Code web : 836961 


Pour l'utiliser, comme indiqué, il va falloir référencer l’assembly System.Numerics.dl1ll1. 
Rien de plus simple, répétons la procédure pour référencer une assembly et allons 
trouver System.Numerics.dll, comme indiqué à la figure 9.8. 


F 
La Ajouter une référence 


.NET |COM [ Projets [ Parcourir [ Récent 
Filtré pour : .NET Framework 4 Client Profile 


2 


| Nom du composant | Version Runtime Chemin d'acc * 
System.ldentityModel.Selectors 4.0.0.0 v4.0,30319 C:\Program F 
System.I0.Log 4.0.0.0 v4.0,30319 C:\Program F 
System.Management 40.00 v4.0,30319 C:\Program F 
System.Management.Instrumentation 40.00 v4.0,30319 C:\Program F 
System.Messaging 40.00 v4.0.30319 C:\Program F 
System.Net 4.0.0.0 v4,0.30319 C:\Program F 


System.Numerics 4.0.0.0 v4.0.30319 C:\Program F 


System.Printing 40.00 v4.0.30319 C:\Program FLal 
System.Runtime.Durablelnstancing 40.00 v4.0.30319 C:\Program F 
System.Runtime.Remoting 40.00 v4.0.30319 C:\Program F 
System.Runtime.Serialization 40.00 v4.0.30319 C:\Program F 
System.Runtime.Serialization.Formatters.Soap 40.00 v4.0.30319 CAProgramF | 


« | mt L 


FIGURE 9.8 — Référencer l’assembly System.Numerics 


Ensuite, nous pourrons utiliser par exemple le code suivant : 


Complex c = Complex.One; 

Console.WriteLine(c); 

Console.WriteLine("Partie réelle : " + c.Real); 
Console.WriteLine("Partie imaginaire : " + c.Imaginary); 


Oo Où À & ND 


Console .WriteLine (Complex.Conjugate (Complex. 
FromPolarCoordinates(1.0, 45 *x Math.PI / 180))); 


Sachant qu'il aura fallu bien sûr ajouter : 
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1 | using System.Numerics; 


permettant d'éviter de préfixer Complex par System.Numerics. Mais ça, vous l’aviez 
trouvé tout seul, n’est-ce pas ? 


(GE (0) 
Partie réelle : 1 


Partie imaginaire : © 
(0,707106781186548, -0,707106781186547) 


En résumé 


— Le framework .NET est un ensemble d’assemblys qu'il faut référencer dans son projet 
afin d’avoir accès à leurs fonctionnalités. 

— On utilise le mot clé using pour inclure un espace de nom comme raccourci dans 
son programme, ce qui permet de ne pas avoir à préfixer les types de leurs espaces 
de noms complets. 
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sie 10 | 


TP : bonjour c est le week-end | 


Difficulté : 16 


ienvenue dans ce premier TP ! Vous avez pu découvrir dans les chapitres précédents 
les premières bases du langage C# permettant la construction d'applications. Il est 
grand temps de mettre en pratique ce que nous avons appris. C'est ici l'occasion pour 
vous de tester vos connaissances et de valider ce que vous appris en réalisant cet exercice. 


C# 
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Instructions pour réaliser le TP 


Le but est de créer une petite application qui affiche un message différent en fonction 
du nom de l'utilisateur et du moment de la journée : 


— « Bonjour X » pour la tranche horaire 9h > 18h, les lundis, mardis, mercredis, jeudis 
et vendredis ; 

— « Bonsoir X » pour la tranche horaire 18h > 9h, les lundis, mardis, mercredis, jeudis ; 

— « Bon week-end X » pour la tranche horaire vendredi 18h > lundi 9h. 


Peut-être cela vous paraît-il simple, dans ce cas, foncez et réalisez cette première ap- 
1 ? 

plication tout seuls. Sinon, décortiquons un peu l’énoncé de ce TP pour éviter d’être 

perdus! 


Pour réaliser ce premier TP, vous allez avoir besoin de plusieurs choses. Dans un premier 
temps, il faut afficher le nom de l'utilisateur, c’est une chose que nous avons déjà faite 
en allant puiser dans les fonctionnalités du framework .NET. 


Vous aurez besoin également de récupérer l’heure courante pour la comparer aux 
tranches horaires souhaitées. Vous avez déjà vu comment récupérer la date courante. 
Pour pouvoir récupérer l'heure de la date courante, il vous faudra utiliser l’instruction 
DateTime.Now.Hour qui renvoie un entier représentant l’heure du jour. 


Pour comparer l’heure avec des valeurs entières il vous faudra utiliser les opérateurs de 
comparaison et les instructions conditionnelles que nous avons vues précédemment. 


Pour traiter le cas spécial du jour de la semaine, vous aurez besoin que le framework 
.NET vous indique quel jour nous sommes. C’est le rôle de l'instruction 
DateTime.Now.DayOfWeek qui est une énumération indiquant le jour de la semaine.Les 
différentes valeurs sont consultables via le code web suivant : 


Doc DayOfWeek 
Code web : 921325 


Pour plus de clarté, nous les reprenons ici : 


Valeur Traduction 
Sunday Dimanche 
Monday Lundi 
Tuesday Mardi 
Wednesday | Mercredi 
Thursday Jeudi 

Friday Vendredi 
Saturday Samedi 


Il ne restera plus qu’à comparer deux valeurs d’énumération, comme on l’a vu dans le 
chapitre sur les énumérations. 


Voilà, vous avez tous les outils en main pour réaliser ce premier TP! N'hésitez pas 
à revenir sur les chapitres précédents si vous avez un doute sur la syntaxe ou sur les 
instructions à réaliser. On ne peut pas apprendre un langage par cœur du premier coup. 
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À vous de jouer! 


Correction 


Vous êtes autorisés à lire cette correction uniquement si vous vous êtes arraché les 
cheveux sur ce TP! Je vois qu’il vous en reste, encore un effort! Si vous avez réussi 
avec brio le TP, vous pourrez également comparer votre travail au mien. 


Quoi qu’il en soit, voici la correction que je propose. Bien évidemment, il peut y en 
avoir plusieurs, mais elle contient les informations nécessaires pour la réalisation de ce 


PP: 


Première chose à faire : créer un projet de type console. J’ai ensuite ajouté le code 
suivant : 


1| static void Main(string[] args) 


2! { 
3 if (DateTime.Now.Day0OfWeek == Day0OfWeek.Saturday || 
DateTime.Now.DayOfWeek == Day0OfWeek.Sunday) 

4 { 

5 // nous sommes le week-end 

6 AfficherBonWeekEnd(); 

7 } 

8 else 

9 { 

10 // nous sommes en semaine 

11 

12 if (DateTime.Now.DayOfWeek == Day0OfWeek.Monday && 
DateTime.Now.Hour < 9) 

13 { 

14 // nous sommes le lundi matin 

15 AfficherBonWeekEnd() ; 

16 } 

17 else 

18 Ÿ 

19 if (DateTime.Now.Hour >= 9 && DateTime.Now.Hour < 

18) 

20 { 

21 // nous sommes dans la journée 

22 AfficherBonjour(); 

23 } 

24 else 

25 { 

26 // nous sommes en soirée 

27 

28 if (DateTime.Now.Day0OfWeek == Day0fWeek.Friday 

&& DateTime.Now.Hour >= 18) 

29 { 

30 // nous sommes le vendredi soir 

31 AfficherBonWeekEnd); 
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32 } 

33 else 

34 { 

35 AfficherBonsoir(); 

36 } 

37 } 

38 } 

39 x 

40! } 

A1 

42| static void AfficherBonWeekEnd() 

43| { 

44 Console.WriteLine("Bon week-end " + Environment .UserName); 
45| } 

46 

47 | static void AfficherBonjour() 

4a8| { 

49 Console.WriteLine("Bonjour " + Environment . UserName) ; 
50| } 

51 

52| static void AfficherBonsoir() 

53| { 

54 Console.WriteLine('"Bonsoir " + Environment .UserName); 
55| } 


Détaillons un peu ce code : 


Au chargement du programme (méthode Main) nous faisons les comparaisons adé- 
quates. Dans un premier temps, nous testons le jour de la semaine de la date courante 
(DateTime.Now.DayOfWleek) et nous la comparons aux valeurs représentant Samedi 
et Dimanche. Si c’est le cas, alors nous appelons une méthode qui affiche le message 
« Bon week-end » avec le nom de l’utilisateur courant que nous pouvons récupérer avec 
Environment .UserName. 


Si nous ne sommes pas le week-end, alors nous testons l’heure de la date courante avec 
DateTime.Now.Hour. Si nous sommes le lundi matin avant 9h, alors nous continuons 
à afficher « Bon week-end ». Sinon, si nous sommes dans la tranche horaire 9h — 18h 
alors nous pouvons appeler la méthode qui affiche « Bonjour ». Dans le cas contraire, il 
reste juste à vérifier que nous ne sommes pas vendredi soir, qui fait partie du week-end, 
sinon on peut afficher le message « Bonsoir ». 


Et voilà, un bon exercice pour manipuler les conditions et les énumérations... 


Aller plus loin 


Ce TP n’était pas très compliqué, il nous a permis de vérifier que nous avions bien 
compris le principe des if et que nous savions appeler des éléments du framework 


.NET. 
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Nous aurions pu simplifier l’écriture de l’application en compliquant en peu les tests 
avec une combinaison de conditions. Par exemple, on pourrait avoir : 


1| if (DateTime.Now.DayOfWeek == Day0OfWeek.Saturday || 
2 DateTime.Now.DayOfWeek == Day0OfWeek.Sunday || 
(DateTime.Now.DayOfWeek == Day0OfWeek.Monday && DateTime.Now 

.Hour < 9) || 

4 (DateTime.Now.DayOfWeek == Day0OfWeek.Friday && DateTime.Now 
.Hour >= 18)) 

5| { 

6 // nous sommes le week-end 

7 AfficherBonWeekEnd() ; 

8| } 

9| else 

10! 

11 // nous sommes en semaine 

12 if (DateTime.Now.Hour >= 9 && DateTime.Now.Hour < 18) 

13 + 

14 // nous sommes dans la journée 

15 AfficherBonjour(); 

16 } 

17 else 

18 { 

19 AfficherBonsoir (); 

20 } 

21| } 


Le premier test permet de vérifier que nous sommes soit samedi, soit dimanche, soit 
que nous sommes lundi et que l’heure est inférieure à 9, soit que nous sommes vendredi 
et que l’heure est supérieure à 18. Nous avons, pour ce faire, combiné les tests avec 
l'opérateur logique « OÙ » : ||. Remarquons que les parenthèses nous permettent d’agir 
sur l’ordre d'évaluation des conditions. Pour que ce soit le week-end, il faut bien sûr être 
« vendredi et que l’heure soit supérieure à 18 » ou « lundi et que l’heure soit inférieure 
à 9 > ou samedi ou dimanche. 


On pourrait encore simplifier en évitant de solliciter à chaque fois le framework .NET 
pour obtenir la date courante. Il suffit de stocker la date courante dans une variable de 
type DateTime. Ce qui donnerait : 


1| DateTime dateCourante = DateTime.Now; 

2 

3| if (dateCourante.DayO0OfWeek == DayOfWeek.Saturday || 

4 dateCourante.DayOfWeek == Day0OfWeek.Sunday || 

5 (dateCourante.Day0fWeek == Day0OfWeek.Monday && dateCourante 
.Hour < 9) || 

6 (dateCourante.Day0fWeek == Day0OfWeek.Friday && dateCourante 
.Hour >= 18)) 

7| € 

8 // nous sommes le week-end 

9 AfficherBonWeekEnd() ; 

10! } 
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11| else 

12 | À 

13 // nous sommes en semaine 

14 if (dateCourante.Hour >= 9 && dateCourante.Hour < 18) 
15 { 

16 // nous sommes dans la journée 
17 AfficherBonjour(); 

18 1 

19 else 

20 { 

21 AfficherBonsoir(); 

22 k 

23| } 


On utilise ici le type DateTime comme le type string ou int. Il sert à gérer les dates 
et l'heure. Il est légèrement différent des types que nous avons vus pour l’instant, nous 
ne nous attarderons pas dessus. Nous aurons l’occasion de découvrir de quoi il s’agit 
dans la partie suivante. Cette optimisation n’a rien d’extraordinaire, mais cela nous 
évite un appel répété au framework .NET. 


Voilà pour ce TP. J'espère que vous aurez réussi avec brio l’exercice. Vous avez pu 
remarquer que ce n’était pas trop difficile. Il a simplement fallu réfléchir à comment 
imbriquer correctement nos conditions. 


N'hésitez pas à pratiquer et à vous entraîner avec d’autres problèmes de votre cru. Si 
vous avez la moindre hésitation, vous pouvez relire les chapitres précédents. Vous verrez 
que nous aurons l’occasion d’énormément utiliser ces instructions conditionnelles dans 
tous les programmes que nous allons écrire! 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code 
Code web : 106683 
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se À 1 


Les boucles 


Difficulté : 1 


ous les avons évoquées rapidement un peu plus tôt en parlant des tableaux et des 
listes. Dans ce chapitre nous allons décrire plus précisément les boucles. 


Elles sont souvent utilisées pour parcourir des éléments énumérables, comme le sont les 


tableaux ou les listes. Elles peuvent également être utilisées pour effectuer la même action 
tant qu'une condition n'est pas réalisée. 


© 


© © 
© 
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La boucle for 


La première instruction que nous avons aperçue est la boucle for. Elle permet de 
répéter un bout de code tant qu’une condition est vraie. Souvent cette condition est un 
compteur. Nous pouvons par exemple afficher un message 50 fois avec le code suivant : 


1| int compteur; 

2| for (compteur = 0; compteur < 50; compteur++) 
3| 

4 Console.WriteLine("Bonjour C#"); 

5|} 


Ce qui affichera : 


C# 
C# 
C# 
fois en tout, mais abrégé pour économiser du papier... 


C# 
C# 
C# 


Nous définissons ici un entier compteur. Il est initialisé à 0 en début de boucle (compteur 
= 0). Après chaque exécution du bloc de code du for, c’est-à-dire à chaque itération, 
il va afficher « Bonjour C# ». À la fin de chaque itération, la variable compteur est 
incrémentée (compteur++) et nous recommençons l'opération tant que la condition 
« compteur < 50 » est vraie. 


Bien sûr, la variable compteur est accessible dans la boucle et change de valeur à chaque 
itération. 


1| int compteur; 

2| for (compteur = 0; compteur < 50; compteur++) 

3| 

4 Console.WriteLine("Bonjour C# " + compteur); 
5|} 


Le code précédent affichera la valeur de la variable après chaque bonjour, donc de 0 à 
49. En effet, quand compteur passe à 50, la condition n’est plus vraie et on passe aux 
instructions suivantes : 


Bonjour C# 
Bonjour C# 
Bonjour C# 


Bonjour C# 

[... abrégé également ...] 
Bonjour C# 48 

Bonjour C# 49 
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En général, on utilise la boucle for pour parcourir un tableau. Aïnsi, nous pourrons 
utiliser le compteur comme indice pour accéder aux éléments du tableau : 


1] string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| int indice; 

3| for (indice = 0; indice < 7; indice++) 

al { 

5 Console.WriteLine(jourslindicel]); 

6| } 


Dans cette boucle, vu qu’à la première itération indice vaut 0, nous afficherons l’élé- 
ment du tableau à la position 0, à savoir « Lundi ». À l’itération suivante, indice passe 
à 1, nous affichons l’élément du tableau à la position 1, à savoir « Mardi », et ainsi de 
suite : 


Lundi 
Mardi 
Mercredi 
Jeudi 
Vendredi 
Samedi 


Dimanche 


Attention à ce que l’indice ne dépasse pas la taille du tableau, sinon l’accès à un indice 
en dehors des limites du tableau provoquera une erreur à l’exécution. Pour éviter ceci, 
on utilise en général la taille du tableau comme condition de fin : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| int indice; 

3| for (indice = 0; indice < jours.Length; indice++) 

al { 

5 Console.WriteLine(jourslindicel]); 

6| } 


Ici jours.Length renvoie la taille du tableau, à savoir 7. 


Il est très courant de boucler sur un tableau en passant tous les éléments un par un, 
mais il est possible de changer les conditions de départ, les conditions de fin, et l’élément 
qui influe sur la condition de fin. 


Ainsi, l'exemple suivant : 


1| int[]l chiffres = new int[] { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; 
2 

3| for (int i = 9; i > O0; i -= 2) 

al { 

5 Console.WriteLine(chiffres[il]); 

6| } 


Permet de parcourir le tableau de 2 en 2 en commençant par la fin : 
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= © O1 = KO 


Vous avez pu voir que dans cet exemple, nous avons défini l’entier i directement dans 
l'instruction for. C’est une commodité qui permet que la variable i n'existe qu’à 
l’intérieur de la boucle, car sa portée correspond aux accolades qui délimitent le for. 


Attention, il est possible de faire un peu tout et n’importe quoi dans ces boucles. Aussi 
il peut arriver que l’on se retrouve avec des bugs, comme des boucles infinies. 


Par exemple, le code suivant : 


1] for (int indice = 0; indice < 7; indice--) 
2! À 

3 Console.WriteLine("Test'" + indice); 

al } 


est une boucle infinie. En effet, on modifie la variable indice en la décrémentant. Sauf 
que la condition de sortie de la boucle est valable pour un indice qui dépasse ou égale 
la valeur 7, ce qui n’arrivera jamais. 


Si on exécute l’application avec ce code, la console va afficher à l’infini le mot « Test » 
avec son indice. La seule solution pour quitter le programme sera de fermer brutalement 
l'application. 


L'autre solution est d'attendre que le programme se termine. .…. 


Q Mais tu viens de dire que la boucle était infinie ? 


Oui c’est vrai, mais en fait, ici on se heurte à un cas limite du C#. C’est à cause de 
la variable indice, qui est un entier que l’on décrémente. Au début il vaut zéro, puis 
-1, puis -2, etc. Lorsque la variable indice arrive à la limite inférieure que le type 
int est capable de gérer, c’est-à-dire -2147483648, se produit alors ce qu’on appelle 
un dépassement de capacité. Sans rentrer dans les détails, il ne peut pas stocker 
un entier plus petit et donc il boucle et repart à l’entier le plus grand, c’est-à-dire 
2147483647. 


Donc pour résumer, l’indice fait : 


-0 
mai 

. 

— 2147483647 
— 2147483648 
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— +2147483647 


Et comme là, il se retrouve supérieur à 7, la boucle se termine! 


0 
-1 
-2 


abrégé ...] 
-2147483647 
-2147483648 


Donc, techniquement, ce n’est pas une boucle infinie, mais on à attendu tellement 
longtemps pour atteindre ce cas limite que c’est tout comme. 


Et surtout, nous tombons sur un cas imprévu. Ici, ça se termine plutôt bien, mais ça 
aurait pu finir en crash de l’application. 


Dans tous les cas, il faut absolument maîtriser ses conditions de sortie de 
boucle pour éviter la boucle infinie, un des cauchemars du développeur | 


La boucle foreach 


Comme il est très courant d’utiliser les boucles pour parcourir un tableau ou une liste, 
le C# dispose d’un opérateur particulier : foreach que l’on peut traduire par « pour 
chaque » ; pour chaque élément du tableau faire ceci... 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }: 

2| foreach (string jour in jours) 

3| À 

4 Console.WriteLine (jour); 

5| } 


Cette boucle va nous permettre de parcourir tous les éléments du tableau jours. À 
chaque itération, la boucle va créer une chaîne de caractères jour qui contiendra l’élé- 
ment courant du tableau. À noter que la variable jour aura une portée égale au bloc 
foreach. Nous pourrons ainsi utiliser cette valeur dans le corps de la boucle et pourquoi 
pas l'afficher comme dans l’exemple précédent : 


Lundi 
Mardi 
Mercredi 
Jeudi 
Vendredi 
Samedi 


Dimanche 
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L’instruction foreach fonctionne aussi avec les listes. Par exemple le code suivant : 


1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" Ÿ}; 

2| foreach (string jour in jours) 

3| À 

4 Console.WriteLine(jour); 

5| } 


nous permettra d’afficher tous les jours de la semaine contenus dans la liste des jours. 


Attention, la boucle foreach est une boucle en lecture seule. Cela veut dire qu’il n’est 
pas possible de modifier l’élément de l’itération en cours. 


Par exemple, le code suivant : 


1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| foreach (string jour in jours) 

3| À 

4 jour = "pas de jour !"; 

5| } 


provoquera l'erreur de compilation suivante : 


Impossible d’assigner à ?’jour”’, car il s’agit d’un variable d? 


itération foreach”? 


Notez d’ailleurs que l’équipe de traduction de Visual C# Express a quelques progrès à 
faire. .. | 


Si nous souhaitons utiliser une boucle pour changer la valeur de notre liste ou de notre 
tableau, il faudra passer par une boucle for : 


1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| for (int i = 0; i < jours.Count; it++ ) 

3| + 

4 joursli] = "pas de jour !"; 

5[} 


Il vous arrivera aussi sûrement un jour (ça arrive à tous les développeurs) de vouloir 
modifier une liste en elle-même lors d’une boucle foreach. C'est-à-dire lui rajouter 
un élément, en supprimer un autre, etc. C’est également impossible, car à partir du 
moment où l’on rentre dans la boucle foreach, la liste devient non-modifiable. 


Prenons l’exemple, ê combien classique, de la recherche d’une valeur dans une liste 
pour la supprimer. Nous serions tentés de faire : 


1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| foreach (string jour in jours) 

3| À 
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if (jour == "Mercredi") 
jours.Remove(jour) ; 


a À 


6| } 


ce qui semblerait tout à fait correct et en plus, ne provoque pas d’erreur de compilation. 
Sauf que si vous exécutez l’application, vous aurez un message d’erreur : 


Exception non gérée : System.InvalidOperationException: La 
collection a été modifiée ; l’opération d’énumération peut ne 
pas s’exécuter. 
System.ThrowHelper.ThrowInvalidOperationException( 
ExceptionResource resource) 


System.Collections.Generic.List ‘1.Enumerator.MoveNextRare() 

System.Collections.Generic.List ‘1. Enumerator.MoveNext () 

MaPremiereApplication.Program.Main(String[]l args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
14 


Le programme nous indique alors que « la collection a été modifiée » et que « l'opération 
d’énumération peut ne pas s’exécuter ». 


Il est donc impossible de faire notre suppression ainsi. 


Comment tu ferais toi ? 


Eh bien, plusieurs solutions existent. Celle qui vient en premier à l'esprit est d’utiliser 
une boucle for par exemple : 


for (int i = 0 ; i < jours.Count ; i++) 
{ 
if (joursli]== "Mercredi") 
jours.Remove(jours[i]l); 


oO & & ND 


} 


Cette solution est intéressante ici, mais elle peut poser un problème dans d’autres 
situations. En effet, vu que nous supprimons un élément de la liste, nous allons nous 
retrouver avec une incohérence entre l’indice en cours et l’élément que nous essayons 
d'atteindre. En effet, lorsque le jour courant est mercredi, l'indice i vaut 2. Si l’on 
supprime cet élément, c’est jeudi qui va se retrouver en position 2. Et nous allons rater 
son analyse car la boucle va continuer à l’indice 3, qui sera vendredi. On peut éviter ce 
problème en parcourant la boucle à l’envers : 


1| for (int i = jours.Count - 1; i >= O0; i--) 
2| { 

3 if (jourslil == "Mercredi") 

4 jours.Remove(jours[il); 

5| } 
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D'une manière générale, il ne faut pas modifier les collections que nous 
sommes en train de parcourir. 


Nous n’étudierons pas les autres solutions car elles font appel à des notions que nous 
verrons en détail plus tard. 


Après avoir lu ceci, vous pourriez avoir l'impression que la boucle foreach n’est pas 
souple et difficilement exploitable : autant utiliser autre chose... Vous verrez à l’utili- 
sation que non, elle est en fait très pratique. Il faut simplement connaître ses limites. 
Voilà qui est chose faite! 


Nous avons vu que l'instruction foreach permettait de boucler sur tous les éléments 
d’un tableau ou d’une liste. En fait, il est possible de parcourir tous les types qui sont 
énumérables. Nous verrons plus loin ce qui caractérise un type énumérable, car pour 
l'instant, c’est un secret ! Chuuut... 


La boucle while 


La boucle while est en général moins utilisée que for ou foreach. C’est la boucle qui 
va nous permettre de faire quelque chose tant qu’une condition n’est pas vérifiée. Elle 
ressemble de loin à la boucle for, mais la boucle for se spécialise dans le parcours de 
tableau tandis que la boucle while est plus générique. 


Par exemple : 

int i = 0; 
while (i < 50) 
{ 


Console.WriteLine("Bonjour C#"); 
Les 


OO Où À À NN 


} 


Ce code permet d'écrire « Bonjour C# » 50 fois de suite. Ici, la condition du while est 
évaluée en début de boucle. 
Dans l’exemple suivant : 
int i = O0; 
do 
{ 
Console.WriteLine("Bonjour C#"); 
Les 
} 
while (i < 50); 


NN OO Où À © NN 


La condition de sortie de boucle est évaluée à la fin de la boucle. L'utilisation du mot-clé 
do permet d’indiquer le début de la boucle. 


Concrètement cela veut dire que dans le premier exemple, le code à l’intérieur de la 
boucle peut ne jamais être exécuté, si par exemple l’entier i est initialisé à 50. À 
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contrario, on passera au moins une fois dans le corps de la boucle du second exemple, 
même si l’entier i est initialisé à 50 car la condition de sortie de boucle est évaluée à 
la fin. 


Les conditions de sortie de boucles ne portent pas forcément sur un compteur ou un 
indice, n'importe quelle expression peut être évaluée. Par exemple : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2! int i = O0; 

3| bool trouve = false; 

4| while (!trouve) 

51 { 

6 string valeur = jourslil; 

7 if (valeur == "Jeudi") 

8 { 

9 trouve = true; 

10 } 

11 else 

12 { 

13 i++; 

14 x 

151 } 

16| Console.WriteLine ("Trouvé à l'indice " + i); 


Le code précédent va répéter les instructions contenues dans la boucle while tant 
que le booléen trouve sera faux (c’est-à-dire qu’on va s’arrêter dès que le booléen sera 
vrai). Nous analysons la valeur pour l'indice i. Si la valeur est celle cherchée, alors nous 
passons le booléen à vrai et nous pourrons sortir de la boucle. Sinon, nous incrémentons 
l'indice pour passer au suivant. 


Attention encore aux valeurs de sortie de la boucle. Si nous ne trouvons pas la chaîne 
recherchée, alors i continuera à s’incrémenter ; le booléen ne passera jamais à vrai, nous 
resterons bloqués dans la boucle et nous risquons d’atteindre un indice qui n’existe pas 
dans le tableau. Il serait plus prudent que la condition porte également sur la taille du 
tableau, par exemple : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2! int i = O0; 

3| bool trouve = false; 

al while (i < jours.Length && !trouve) 

51 { 

6 string valeur = jourslil; 

7 if (valeur == "Jeudi") 

8 { 

9 trouve = true; 

10 } 

11 else 

12 { 

13 i++; 


99 


CHAPITRE 11. LES BOUCLES 


14 ; 

15 | } 

16| if (!trouve) 

17 Console.WriteLine("Valeur non trouvée"); 

18 | else 

19 Console.WriteLine ("Trouvé à l'indice " + i); 


Ainsi, si l’indice est supérieur à la taille du tableau, nous sortons de la boucle et nous 
éliminons le risque de boucle infinie. 


Une erreur classique est que la condition ne devienne jamais vraie à cause d’une erreur 
de programmation. Par exemple, si j'oublie d’incrémenter la variable i, alors à chaque 
passage de la boucle j’analyserai toujours la première valeur du tableau et je n’atteindrai 
jamais la taille maximale du tableau; condition qui me permettrait de sortir de la 
boucle. 


À Je ne le répéterai jamais assez : faites bien attention aux conditions de sortie 


d'une boucle! 


Les instructions break et continue 


Il est possible de sortir prématurément d’une boucle grâce à l’instruction break. Dès 
qu’elle est rencontrée, elle sort du bloc de code de la boucle. L’exécution du programme 
continue alors avec les instructions situées après la boucle. Par exemple : 


1| int i = GO; 

2| while (true) 

3| { 

4 if (i >= 50) 

5 { 

6 break; 

7 } 

8 Console.WriteLine("Bonjour C#"); 
9 i++; 

10 | } 


Le code précédent pourrait potentiellement produire une boucle infinie. En effet, la 
condition de sortie du while est toujours vraie. Mais l’utilisation du mot-clé break 
nous permettra de sortir de la boucle dès que i atteindra la valeur 50. Certes ici, il 
suffirait que la condition de sortie porte sur l'évaluation de l’entier i. Mais il y a des 
cas où il pourra être judicieux d’utiliser un break (surtout lors d’un déménagement !). 


C’est le cas pour l'exemple suivant. Imaginons que nous voulions vérifier la présence 
d’une valeur dans une liste. Pour la trouver, on peut parcourir les éléments de la liste 
et une fois trouvée, on peut s’arrêter. En effet, il serait inutile de continuer à parcourir 
le reste des éléments : 
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1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" #}; 

2| bool trouve = false; 

3| foreach (string jour in jours) 

al { 

5 if (jour == "Jeudi") 

6 { 

7 trouve = true; 

8 break; 

9 F 

10! } 


Nous nous dispensons ici d'analyser les 3 derniers éléments de la liste. 


Il est également possible de passer à l’itération suivante d’une boucle grâce à l’utilisation 
du mot-clé continue. Prenons l’exemple suivant : 


1] for (int i = O0; i < 20; i++) 
2| { 

3 if (i % 2 == 0) 

4 { 

5 continue; 

6 } 

7 Console.WriteLine(i); 
8| } 


Ici, l'opérateur % est appelé « modulo ». Il permet d’obtenir le reste de la division. 
L'opération i % 2 renverra donc 0 quand i est pair. Ainsi, dès qu’un nombre pair est 
trouvé, nous passons à l’itération suivante grâce au mot-clé continue. Ce qui fait que 
nous n’afficherons que les nombres impairs : 


Bien sûr, il aurait été possible d’inverser la condition du if pour n’exécuter le 
Console.WriteLine() que dans le cas où i % 2!= 0. À vous de choisir l'écriture 
que vous préférez en fonction des cas que vous rencontrez. 


En résumé 


— On utilise la boucle for pour répéter des instructions tant qu’une condition n’est 
pas vérifiée, les éléments de la condition changeant à chaque itération. 
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— On utilise en général la boucle for pour parcourir un tableau, avec un indice qui 
s’incrémente à chaque itération. 

— La boucle foreach est utilisée pour simplifier le parcours des tableaux ou des listes. 

— La boucle while permet de répéter des instructions tant qu’une condition n’est pas 


vérifiée. C’est la boucle la plus souple. 
— Il faut faire attention aux conditions de sortie d’une boucle afin d'éviter les boucles 


infinies qui font planter l’application. 
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TP : calculs en boucle 


Difficulté : 16 


a yest, grâce au chapitre précédent, vous devez avoir une bonne idée de ce que sont 

les boucles. Par contre, vous ne voyez peut-être pas encore complètement qu'elles 

vont vous servir tout le temps. C'est un élément qu'il est primordial de maîtriser. Il 
vous faut donc vous entraîner pour être bien sûrs d'avoir tout compris. C'est justement 
l'objectif de ce deuxième TP. Finie la théorie des boucles, place à la pratique en boucle! 
Notre but est de réaliser des calculs qui vont avoir besoin des for et des foreach. Vous 
êtes prêts ? Alors, c'est parti! 


C# 
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Instructions pour réaliser le TP 


Le but de ce TP va être de créer trois méthodes. 


La première va servir à calculer la somme d’entiers consécutifs. Si par exemple je veux 
calculer la somme des entiers de 1 à 10, c’est-à-dire 1 +2+3+4+5+6+7+8 
9 + 10, je vais appeler cette méthode en lui passant en paramètres 1 et 10, c’est-à-dire 
les bornes des entiers dont il faut faire la somme. 


Quelque chose du genre : 


1| Console.WriteLine (CalculSommeEntiers(1, 10)); 
2| Console.WriteLine (CalculSommeEntiers(1, 100)); 


Sachant que le premier résultat de cet exemple vaut 55 (1+2+3+4+5+6+7 
+ 8 +9 + 10 — 55) et le deuxième 5050. 


La deuxième méthode acceptera une liste de double en paramètres et devra renvoyer 
la moyenne des doubles de la liste. Par exemple : 


1| List<double> liste = new List <double> { 1.0, 5.5, 9.9, 2.8, 9.6 
1 


2| Console.WriteLine (CalculMoyenne (liste)); 


Le résultat de cet exemple vaut 5.76. 


Enfin, la dernière méthode devra dans un premier temps construire une liste d’entiers 
de 1 à 100 qui sont des multiples de 3 (3, 6, 9, 12,...). Dans un second temps, construire 
une autre liste d’entiers de 1 à 100 qui sont des multiples de 5 (5, 10, 15, 20,...). Et 
dans un dernier temps, il faudra calculer la somme des entiers qui sont communs aux 
deux listes... vous devez bien sûr trouver 315 comme résultat ! 


Voilà, c’est à vous de jouer... 


Bon, allez, je vais quand même vous donner quelques conseils pour démarrer. Vous 
n'êtes pas obligés de les lire si vous vous sentez capables de réaliser cet exercice tout 
seul. 


Vous l’aurez évidemment compris, il va falloir utiliser des boucles. Je ne donnerai pas 
de conseils pour la première méthode, c’est juste pour vous échauffer ! 


Pour la deuxième méthode, vous allez avoir besoin de diviser la somme de tous les 
double par la taille de la liste. Vous ne savez sans doute pas comment obtenir cette 
taille. Le principe est le même que pour la taille d’un tableau et vous l’aurez sans doute 
trouvé si vous fouillez un peu dans les méthodes de la liste. Toujours est-il que pour 
obtenir la taille d’une liste, on va utiliser liste.Count, avec par exemple : 


1 | int taille = liste.Count; 
Enfin, pour la dernière méthode, vous allez devoir trouver tous les multiples de 3 et 
de 5. Le plus simple, à mon avis, pour calculer tous les multiples de 3, est de faire une 


boucle qui démarre à 3 et d'avancer de 3 en 3 jusqu’à la valeur souhaitée. On fait de 
même pour les multiples de 5. 
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CORRECTION 


Ensuite, il sera nécessaire de faire deux boucles imbriquées afin de déterminer les in- 
tersections. C'est-à-dire parcourir la liste des multiples de 3 et à l’intérieur de cette 
boucle, parcourir la liste des multiples de 5. On compare les deux éléments ; s’ils sont 
égaux, c’est qu'ils sont communs aux deux listes. 


Voilà, vous devriez avoir tous les éléments en main pour réussir ce TP, c’est à vous de 
jouer ! 


Correction 


J'imagine que vous avez réussi ce TP, non pas sans difficultés, mais à force de tâton- 
nements, vous avez sans doute réussi. Bravo! Quoi qu’il en soit, voici la correction que 
je propose. 


— Première méthode : 


1| static int CalculSommeEntiers(int borneMin, int borneMax) 
2! { 

à int resulat = O0; 

4 for (int i = borneMin; i <= borneMax ; i++) 

5 { 

6 resulat += ji; 

7 } 

8 return resulat; 

9! } 


Ce n’était pas très compliqué, il faut dans un premier temps construire une méthode qui 
renvoie un entier et qui accepte deux entiers en paramètres. Ensuite, on boucle grâce 
à l'instruction for de la borne inférieure à la borne supérieure (incluse, donc il faut 
utiliser opérateur de comparaison <— ) en incrémentant un compteur de 1 à chaque 
itération. Nous ajoutons la valeur du compteur à un résultat, que nous retournons en 


fin de boucle. 


— Seconde méthode : 


1] static double CalculMoyenne (List <double> liste) 
2! { 

8 double somme = O0; 

4 foreach (double valeur in liste) 

5 { 

6 somme += valeur; 

7 Hs 

8 return somme / liste.Count; 

9! } 


Ici, le principe est grosso modo le même, la différence est que la méthode retourne 
un double et accepte une liste de double en paramètres. Ici, nous utilisons la boucle 
foreach pour parcourir tous les éléments que nous ajoutons à un résultat. Enfin, nous 
retournons ce résultat divisé par la taille de la liste. 
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— Troisième méthode : 


1| static int CalculSommelntersection() 

2| { 

3 List<int> multiplesDe3 = new List<int>(); 
4 List<int> multiplesDe5 = new List<int>(); 
5 

6 for (int i = 3; i <= 100; i += 3) 

7 { 

8 multiplesDe3.Add(i); 

9 } 

10 for (int i = 5; i <= 100; i += 5) 

11 { 

12 multiplesDe5.Add(i); 

13 } 

14 

15 int somme = O; 

16 foreach (int m3 in multiplesDe3) 

17 { 

18 foreach (int m5 in multiplesDe5) 
19 { 

20 if (m3 == m5) 

21 somme += m3; 

22 F 

23 } 

24 return somme; 

25| } 


Cette troisième méthode est peut-être la plus compliquée... 


On commence par créer nos deux listes de multiples. Comme je vous avais conseillé, 
j'utilise une boucle for qui commence à 3 avec un incrément de 3. Comme ça, je 
suis sûr d’avoir tous les multiples de 3 dans ma liste. C’est le même principe pour les 
multiples de 5, sachant que dans les deux cas, la condition de sortie est quand l’indice 
est supérieur à 100. 


Ensuite, j'ai mes deux boucles imbriquées où je compare les deux valeurs et si elles 
sont égales, je rajoute la valeur à la somme globale que je renvoie en fin de méthode. 
Pour bien comprendre ce qu’il se passe dans les boucles imbriquées, il faut comprendre 
que nous allons parcourir une unique fois la liste multiplesDe3 mais que nous al- 
lons parcourir autant de fois la liste multipleDeS qu’il y a d'éléments dans la liste 
multipleDes, c’est-à-dire 33 fois. Ce n’est sans doute pas facile de le concevoir dès le 
début, mais pour vous aider, vous pouvez essayer de vous faire l’algorithme dans la 
tête : 


— on rentre dans la boucle qui parcoure la liste multiplesDe3 ; 
— m3 vaut 3; 

— on rentre dans la boucle qui parcoure la liste multiplesDes ; 
— m5 vaut 5; 

— on compare 3 à 5, ils sont différents ; 

— on passe à l’itération suivante de la liste multiplesDes ; 
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— m5 vaut 10; 

— on compare 3 à 10, ils sont différents ; 

— de même jusqu’à ce qu’on ait fini de parcourir la liste des multiplesDes : 
— on passe à l’itération suivante de la liste multiplesDe3 ; 

— m3 vaut 6; 

— on rentre dans la boucle qui parcoure la liste multiplesDes ; 

— m5 vaut 5; 

— on compare 6 à 5, ils sont différents ; 

— on passe à l’itération suivante de la liste multiplesDes ; 

— etc. 


Aller plus loin 


Vous avez remarqué que dans la deuxième méthode, j'utilise une boucle foreach pour 
parcourir la liste : 


1| static double CalculMoyenne (List <double> liste) 
2! { 

8 double somme = O0; 

4 foreach (double valeur in liste) 

5 € 

6 somme += valeur; 

7 } 

8 return somme / liste.Count; 

9! } 


Il est aussi possible de parcourir les listes avec un for et d'accéder aux éléments de la 
liste avec un indice, comme on le fait pour un tableau. Ce qui donnerait : 


1| static double CalculMoyenne (List <double> liste) 
2! { 

3 double somme = O0; 

4 for (int i = 0; i < liste.Count; i++) 

5 { 

6 somme += listelil; 

7 x 

8 return somme / liste.Count; 

9! } 


Notez qu’on se sert de liste.Count pour obtenir la taille de la liste et qu’on accède à 
l’élément courant avec liste [i]. Je reconnais que la boucle foreach est plus explicite, 
mais cela peut être utile de connaître le parcours d’une liste avec la boucle for. À vous 
de voir. 


Vous aurez également noté ma technique pour avoir des multiples de 3 et de 5. Il y en 
a plein d’autres. Je pense à une en particulier et qui fait appel à des notions que nous 
avons vues auparavant : la division entière et la division de double. 
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Pour savoir si un nombre est un multiple d’un autre, il suffit de les diviser entre eux et 
de voir s’il y a un reste à la division. On peut faire cela de deux façons. La première, 
en comparant la division entière avec la division double et en vérifiant que le résultat 
est le même. Si le résultat est le même, c’est qu’il n’y a pas de chiffres après la virgule : 


1| for (int i = 1; i <= 100; i++) 
2| { 

3 if (i / 3 == i / 3.0) 

4 multiplesDe3.Add(i); 

5 if (i / 5 == i / 5.0) 

6 multiplesDe5.Add(i); 
T|} 


comment se fait-il qu'on puisse comparer un entier à un double ? Nous le 
verrons un peu plus tard. 


Q@ Cette technique fait appel à des notions que nous n'avons pas encore vues : 


L'autre solution est d'utiliser l'opérateur modulo que nous avons vu précédemment qui 
fait justement ça : 


1| for (int i = 1; i <= 100; i++) 
2| € 

3 if (i % 3 == 0) 

4 multiplesDe3.Add(i); 

5 if (Ci % 5 == 0) 

6 multiplesDe5.Add(i); 
T|} 


L'avantage de ces deux techniques est qu’on peut construire les deux listes de multiples 
en une seule boucle. 


Voilà, c’est fini pour ce TP. N'hésitez pas à vous faire la main sur ces boucles car il est 
fondamental que vous les maîtrisiez. Faites-vous plaisir, tentez de les repousser dans 
leurs limites, essayez de résoudre d’autres problèmes... L'important, c’est de pratiquer. 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 


vant : 


Copier ce code 
Code web : 764808 
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Deuxième partie 


Un peu plus loin avec le C 


109 


sie 1 3 | 


Les conversions entre les types 


Difficulté : 3 


ous avons vu que le C# est un langage qui possède plein de types de données 
différents : entier, décimal, chaîne de caractères, etc. 


Dans nos programmes, nous allons très souvent avoir besoin de manipuler des données 


entre elles alors qu'elles ne sont pas forcément du même type. Lorsque cela sera possible, 
nous aurons besoin de convertir un type de données en un autre. 
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CHAPITRE 13. LES CONVERSIONS ENTRE LES TYPES 


Entre les types compatibles : le casting 


C’est l’heure de faire la sélection des types les plus performants, place au casting ! Le 
jury est en place, à vos SMS pour voter. 


Ab, on me fait signe qu’il ne s’agirait pas de ce casting-là... 


Q Le casting est simplement l’action de convertir la valeur d'un type dans un 


autre. 


Plus précisément, cela fonctionne pour les types qui sont compatibles entre eux, enten- 
dez par là « qui se ressemblent ». 


Par exemple, l’entier et le petit entier se ressemblent. Pour rappel, nous avons : 


Type | Description 
short | Entier de -32768 à 32767 
int Entier de -2147483648 à 2147483647 


Ils ne diffèrent que par leur capacité. Le short ne pourra pas contenir le nombre 100000 
par exemple, alors que l’int le pourra. 


Nous ne l’avons pas encore fait, mais le C# nous autorise à faire : 
1] short s = 200; 


2| int i = s; 


En effet, il est toujours possible de stocker un petit entier dans un grand. Peu importe 
la valeur de s, i sera toujours à même de contenir sa valeur. C’est comme dans une 
voiture. Si nous arrivons à tenir à 5 dans un monospace, nous pourrons facilement tenir 
à 5 dans un bus. 


L’inverse n’est pas vrai bien sûr. Si nous tenons à 100 dans un bus, ce n’est pas certain 
que nous pourrons tenir dans le monospace, même en se serrant bien. 


Ce qui fait que si nous faisons : 


1| int i = 100000; 
2| short s = i;: 


nous aurons l'erreur de compilation suivante : 


Impossible de convertir implicitement le type ?’int? en ?’short”. 


Une conversion explicite existe (un cast est-il manquant ?) 


Eh oui, comment pouvons-nous faire rentrer 100000 dans un type qui ne peut aller que 
jusqu’à 32767? Le compilateur le sait bien. 


Vous remarquerez que nous aurons cependant la même erreur si nous tentons de faire : 
1| int i = 200; 


2| short s = ji; 
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Mais pourquoi ? Le short est bien capable de stocker la valeur 200 ? 


© 


Oui tout à fait, mais le compilateur nous avertit quand même. Il nous dit : 


À « Attention, vous essayez de faire rentrer les personnes du bus dans un mo- 


nospace, êtes-vous bien sûr? » 


Nous avons envie de lui répondre : 


Q « Oui, oui, je sais qu'il y a très peu de personnes dans le bus et qu'ils pourront 


rentrer sans aucun problème dans le monospace. Fais-moi confiance! » 


Eh bien, ceci s’écrit en CZ de la manière suivante : 


1| int i = 200; 
2| short s = (short)i; 


Nous utilisons ce qu'on appelle un cast explicite. 


Pour faire un tel cast, il suffit de faire précéder la variable à convertir du nom du 
type souhaité entre parenthèses. Cela permet d’indiquer à notre compilateur que nous 
savons ce que nous faisons, et que l’entier tiendra correctement dans le petit entier. 


Mais attention, le compilateur nous fait confiance. Nous sommes le boss ! Cela implique 
une certaine responsabilité, il ne faut pas faire n'importe quoi. 


Si nous faisons : 


int i = 40000; 

short s = (short)i; 
Console.WriteLine(s); 
Console.WriteLine(i); 


& © ND + 


nous tentons de faire rentrer un trop gros entier dans ce qu'est capable de stocker le 
petit entier. 


Si nous exécutons notre application, nous aurons un résultat surprenant : 


-25536 
40000 
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Le résultat n’est pas du tout celui attendu. Nous avons fait n’importe quoi, le C# nous 
a punis! 


En fait, plus précisément, il s’est passé ce qu’on appelle un dépassement de capacité. 
Nous l’avons déjà vu lors du chapitre sur les boucles. Ici, il s’est produit la même chose. 
Le petit entier est allé à sa valeur maximale puis a bouclé en repartant de sa valeur 
minimale. 


Bref, tout ça pour dire que nous obtenons un résultat inattendu. Il faut donc faire 
attention à ce que l’on fait et honorer la confiance que nous porte le compilateur en 
faisant bien attention à ce que l’on caste. !. 


D'une façon similaire à l’entier et au petit entier, l’entier et le double se ressemblent un 
peu. Ce sont tous les deux des types permettant de contenir des nombres. Le premier 
permet de contenir des nombres entiers, le deuxième peut contenir des nombres à 
virgules. 


Ainsi, nous pouvons écrire le code C# suivant : 
1| int À = 20; 


2| double d = i; 


En effet, un double est plus précis qu’un entier. Il est capable d’avoir des chiffres après 
la virgule alors que l’entier ne le peut pas. Ce qui fait que le double est entièrement 
capable de stocker toute la valeur d’un entier sans perdre d’information dans cette 
affectation. 


Par contre, comme on peut s’y attendre, l’inverse n’est pas possible. Le code suivant : 


1| double prix = 125.656; 
2| int valeur = prix; 


produira l’erreur de compilation suivante : 


Impossible de convertir implicitement le type ?double? en ?’int”. 


Une conversion explicite existe (un cast est-il manquant ?) 


En effet, un double étant plus précis qu’un entier, si nous mettons ce double dans 
l’entier nous allons perdre notamment les chiffres après la virgule. 


Il restera encore une fois à demander au compilateur de nous faire confiance : « OK, 
ceci est un double, mais ce n’est pas grave, je veux l'utiliser comme un entier ! Oui, oui, 
même si je sais que je vais perdre les chiffres après la virgule ». 


Ce qui s'écrit en C# de cette manière, comme nous l’avons vu : 
1] double prix = 125.56; 


2| int valeur = (int)prix; // valeur vaut 125 


Nous faisons précéder la variable prix du type dans lequel nous voulons la convertir 
en utilisant les parenthèses. 


1. Nous utiliserons souvent cet anglicisme, qui signifiera bien sûr que nous convertissons des types 
qui se ressemblent entre eux, pour le plus grand malheur des professeurs de français. 
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En l'occurrence, on peut se servir de ce cast pour récupérer la partie entière 
d'un nombre à virgule. 


C’est plutôt sympa comme instruction. Mais n’oubliez pas que cette opération est 
possible uniquement avec les types qui se ressemblent entre eux. 


Par exemple, le cast suivant est impossible : 


1] string nombre = "123"; 
2| int valeur = (int)nombre; 


car les deux types sont trop différents et sont incompatibles entre eux. Même si la 
chaîne de caractères représente un nombre. 


Nous verrons plus loin comment convertir une chaîne en entier. Pour l'instant, Visual 
C# Express nous génère une erreur de compilation : 


Impossible de convertir le type string?’ en ’int”? 


Le message d'erreur est plutôt explicite. Il ne nous propose même pas d'utiliser 
de cast, il considère que les types sont vraiment trop différents! 


Nous avons vu précédemment que les énumérations représentaient des valeurs entières. 
Il est donc possible de récupérer l’entier correspondant à une valeur de l’énumération 
grâce à un cast. 


Par exemple, en considérant l’énumération suivante : 


enum Jours 


1 
2! À 

3 Lundi = 5, // lundi vaut 5 

4 Mardi, // mardi vaut 6 

5 Mercredi = 9, // mercredi vaut 9 
6 Jeudi = 10, // jeudi vaut 10 

“A Vendredi, // vendredi vaut 11 

8 Samedi, // samedi vaut 12 

9 Dimanche = 20 // dimanche vaut 20 
10! } 


Le code suivant : 


1 | int valeur = (int)Jours.Lundi; // valeur vaut 5 


convertit l’énumération en entier. 


Nous verrons que le cast est beaucoup plus puissant que ça, mais pour l'instant, nous 
n'avons pas assez de connaissances pour tout étudier. Nous y reviendrons dans la partie 
suivante. 
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Entre les types incompatibles 


Nous avons vu qu'il était facile de convertir des types qui se ressemblent grâce au cast. 
Il est parfois possible de convertir des types qui ne se ressemblent pas mais qui ont le 
même sens. 


Par exemple, il est possible de convertir une chaîne de caractères qui contient unique- 
ment des chiffres en un nombre (entier, décimal, etc.). Cette conversion va nous servir 
énormément car dès qu’un utilisateur va saisir une information par le clavier, elle sera 
représentée par une chaîne de caractères. Donc si on lui demande de saisir un nombre, 
il faut être capable d'utiliser sa saisie comme un nombre afin de le transformer, de le 
stocker, etc. 


Pour ce faire, il existe plusieurs solutions. La plus simple est d’utiliser la méthode 
Convert.Tolnt32() disponible dans le framework .NET. Par exemple : 


1| string chaineAge = "20"; 
2| int age = Convert.Tolnt32(chaineAge); // age vaut 20 


Cette méthode, bien que simple d’utilisation, présente un inconvénient dans le cas où 
la chaîne ne représente pas un entier. À ce moment-là, la conversion va échouer et le 
C+# va renvoyer une erreur au moment de l’exécution. 


Par exemple : 


1] string chaineAge = "vingt ans"; 
2| int age = Convert.Tolnt32(chaineAge); 


Si vous exécutez ce bout de code, vous verrez que la console nous affiche ce que l’on 
appelle une exception : 


Exception non gérée : System.FormatException: Le format de la 
chaîne d’entrée est incorrect. 
System.Number.StringToNumber (String str, NumberStyles options, 
NumberBuffer 


number, NumberFormatIlnfo info, Boolean parseDecimal) 
System.Number.Parselnt32(String s, NumberStyles style, 


NumberFormatInfo info) 
System.Convert.Tolnt32(String value) 
MaPremiereApplication.Program.Main(String[] args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
14 


Il s’agit tout simplement d’une erreur. Nous aurons l’occasion d'étudier plus en détail 
les exceptions dans les chapitres ultérieurs. Pour l’instant, on a juste besoin de voir que 
ceci ne nous convient pas. L'erreur est explicite : « Le format de la chaîne d’entrée est 
incorrect », mais cela se passe au moment de l'exécution et notre programme plante 
lamentablement. Nous verrons dans le chapitre sur les exceptions comment gérer les 
erreurs. 
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En interne, la méthode Convert.Tolnt32() utilise une autre méthode pour faire la 
conversion, il s’agit de la méthode int .Parse(). Donc plutôt que d'utiliser une méthode 
qui en appelle une autre, nous pouvons nous en servir directement, par exemple : 


1] string chaineAge = "20; 
2| int age = int.Parse(chaineAge); // age vaut 20 


Bien sûr, il se passera exactement la même chose que précédemment quand la chaîne 
ne représentera pas un entier correct. 


Il existe par contre une autre façon de convertir une chaîne en entier qui ne provoque 
pas d’erreur quand le format n’est pas correct et qui nous informe si la conversion s’est 
bien passée ou non, c’est la méthode int.TryParse() qui s'utilise ainsi : 


1] string chaiïineÂge = "ab20cd'"; 

2| int age; 

3| if (int.TryParse(chaineAge, out age)) 

al À 

5 Console.WriteLine("La conversion est possible, age vaut " + 
age) ; 

6| } 

7| else 

8| 

9 Console.WriteLine ("Conversion impossible"); 

10| } 


Nous aurons alors : 


Conversion impossible 


La méthode int.TryParse nous renvoie vrai si la conversion est bonne, et faux sinon. 
Nous pouvons donc effectuer une action en fonction du résultat grâce aux if/else. 
On passe en paramètres la chaîne à convertir et la variable où l’on veut stocker le 
résultat de la conversion. Le mot-clé out signifie juste que la variable est modifiée par 
la méthode. 


Les méthodes que nous venons de voir Convert.ToString() ou int.TryParse() se 
déclinent en général pour tous les types de base, par exemple double.TryParse() ou 
Convert.ToDecimal(), etc. 


En résumé 


— Ilest possible, avec le casting, de convertir la valeur d’un type dans un autre lorsqu'ils 
sont compatibles entre eux. 

— Le casting explicite s’utilise en préfixant une variable par un type précisé entre pa- 
renthèses. 

— Le framework .NET possède des méthodes permettant de convertir des types incom- 
patibles entre eux s’ils sont sémantiquement proches. 
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Lire le clavier dans la console 


Difficulté : 1: 


ous avons beaucoup écrit avec Console.WriteLine(), maintenant il est temps de 

savoir lire. En l'occurrence, il faut être capable de récupérer ce que l'utilisateur a saisi 

au clavier. En effet, un programme qui n'a pas d'interaction avec l'utilisateur, ce n'est 
pas vraiment intéressant. Vous imaginez un jeu où on ne fait que regarder ? Ça s'appelle 
une vidéo, c'est intéressant aussi, mais ce n'est pas pareil | 


Aujourd'hui, il existe beaucoup d'interfaces de communication que l'on peut utiliser sur 
un ordinateur (clavier, souris, doigts, manettes... .). Dans ce chapitre nous allons regarder 
comment savoir ce que l'utilisateur a saisi au clavier dans une application console. 


119 


CHAPITRE 14. LIRE LE CLAVIER DANS LA CONSOLE 


Lire une phrase 


Lorsque nous lui en donnerons la possibilité, l’utilisateur de notre programme pourra 
saisir des choses grâce à son clavier. Nous pouvons lui demander son âge, s’il veut 
quitter l’application, lui permettre de saisir un mail, etc. 


Il nous faut donc trouver un moyen lui permettant de saisir des caractères en tapant 
sur son clavier. Nous pourrons faire cela grâce à la méthode Console.ReadLine : 


1 | string saisie = Console.ReadLine(); 


Lorsque notre application rencontre cette instruction, elle se met en pause et attend 
une saisie de la part de l’utilisateur. La saisie s’arrête lorsque l'utilisateur valide ce qu’il 


a écrit avec la touche (Entrée) Ainsi les instructions suivantes : 


1| Console.WriteLine("Veuillez saisir une phrase et valider avec 
la touche \"Entrée\""); 

2| string saisie = Console.ReadLine(); 

3| Console.WriteLine("Vous avez saisi : " + saisie); 


produiront le message suivant : 


Veuillez saisir une phrase et valider avec la touche "Entrée" 
Bonjour à tous 


Vous avez saisi : Bonjour à tous 


Vous aurez remarqué que nous stockons le résultat de la saisie dans une variable de 
type chaîne de caractères. C’est bien souvent le seul type que nous aurons à notre 
disposition lors des saisies utilisateurs. Cela veut bien sûr dire que si vous avez besoin 
de manipuler l’âge de la personne, il faudra la convertir en entier. Par exemple : 


1| bool ageIlsValid = false; 

2| int age = -1; 

3| while (!'agelsValid) 

alt 

5 Console.WriteLine("Veuillez saisir votre âge"); 

6 string saisie = Console.ReadLine() ; 

Ke if (int.TryParse(saisie, out age)) 

8 agelsValid = true; 

9 else 

10 { 

11 ageIlsValid = false; 

12 Console.WriteLine ("L'âge que vous avez saisi est 
lncorreét :::"); 

13 } 

1! } 

15] Console.WriteLine ("Votre âge est de : " + age); 


Le programme insistera alors pour que l'utilisateur entre un nombre : 
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Veuillez saisir votre âge 
abc 
L’âge que vous avez saisi est incorrect... 


Veuillez saisir votre âge 
25 
Votre âge est de : 25 


Ce code est facile à comprendre maintenant que vous maîtrisez les boucles et les condi- 
tions. Nous commençons par initialiser nos variables. Ensuite, nous demandons à l’uti- 
lisateur de saisir son âge. Nous tentons de le convertir en entier. Si cela fonctionne, on 
pourra passer à la suite, sinon, tant que l’âge n’est pas valide, nous recommençons la 
boucle. 


Il est primordial de toujours vérifier ce que saisit l'utilisateur, cela vous évi- 
tera beaucoup d'erreurs et de plantages intempestifs. On dit souvent qu'en 
informatique, il ne faut jamais faire confiance à l'utilisateur ! 


Lire un caractère 


Il peut arriver que nous ayons besoin de ne saisir qu’un seul caractère. Pour cela, nous 
allons pouvoir utiliser la méthode Console .ReadKey(). Nous pourrons nous en servir 
par exemple pour faire une pause dans notre application. Le code suivant réclame 
l’attention de l’utilisateur avant de démarrer un calcul hautement important : 


1| Console.WriteLine("Veuillez appuyer sur une touche pour dé 


marrer Le calcul .::%);: 
2| Console.ReadKey(true); 
3| int somme = O0; 
4| for (int i = 0; i < 100; i++) 
51 { 
6 somme += ji; 
713 
8 


Console.WriteLine (somme); 


Notez que nous avons passé true en paramètre de la méthode afin d’indiquer au C# 
que nous ne souhaitions pas que notre saisie apparaisse à l'écran. Si le paramètre avait 
été false et que j'avais par exemple appuyé sur pour démarrer le calcul, le résultat 
se serait vu précédé d’un disgracieux « H »..…. 


D'ailleurs, comment faire pour savoir quelle touche a été saisie ? 


Il suffit d'observer le contenu de la variable renvoyée par la méthode Console .ReadKey. 
Elle renvoie en l’occurrence une variable du type ConsoleKeyInfo. Nous pourrons 
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tester la valeur Key de cette variable qui est une énumération du type ConsoleKey. Par 
exemple : 


1| Console.WriteLine ("Voulez-vous continuer (O/N) 7"); 
2| ConsoleKeyInfo saisie = Console.ReadKey(true); 

3| if (saisie.Key == ConsoleKey.0) 

al { 

5 Console.WriteLine("On continue ..."); 

6| } 

7| else 

8| { 

9 Console.WriteLine("On s'arrête ..."); 

10! } 


Nous remarquons grâce à la complétion automatique que l’énumération ConsoleKey 
possède plein de valeurs, comme illustré à la figure 14.1. 


static void Main(string[] args) 


{ 
Console.WriteLine("Voulez-vous continuer (O/N) ?"}); 
ConsoleKeyInfo saisie = Console.ReadKey(true); 
if (saisie.Key == Consolekey.|) 
Console.WriteLine("On con 
a” os 
[l I] 
else sŸ Add 
€ sŸ Applications 
Console.WriteLine("On s'a:f Attention 
} sp B 
} sP Backspace 


2? BrowserBack 
2? BrowserFavorites 
2? BrowserForward v 


FIGURE 14.1 - La structure ConsoleKey contient toutes les touches du clavier 


Nous comparons donc à la valeur ConsoleKey.0 qui représente la touche (0). 


Notez que le type ConsoleKeyInfo est ce qu’on appelle une structure et que Key est 
une propriété de la structure. Nous reviendrons dans la partie suivante sur ce que cela 
veut vraiment dire. Pour l'instant, considérez juste qu'il s’agit d’une variable évoluée 
qui permet de contenir plusieurs choses... 


Quand même, je trouve que c’est super de pouvoir lire les saisies de l'utilisateur ! 
C’est vital pour toute application qui se respecte. Aujourd’hui, dans une application 
moderne, il est primordial de savoir lire la position de la souris, réagir à des clics sur 
des boutons, ouvrir des fenêtres, etc. C’est quelque chose qu’il n’est pas possible à faire 
avec les programmes en mode console. N'oubliez pas que ces applications sont parfaites 
pour apprendre le C#, cependant il est inimaginable de réaliser une application digne 
de ce nom avec une application console. 


Pour ce faire, on utilisera des systèmes graphiques adaptés, qui sortent du cadre d’étude 
de ce livre. 
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En résumé 


— La méthode Console.ReadLine nous permet de lire des informations saisies par 
l'utilisateur au clavier. 
— Il est possible de lire toute une phrase terminée par la touche ou seulement 


un unique caractère. 


123 


CHAPITRE 14. LIRE LE CLAVIER DANS LA CONSOLE 


124 


ae D 


Utiliser le débogueur 


Difficulté : 16 


ous allons maintenant faire une petite pause. Le C# c'est bien, mais notre envi- 

ronnement de développement, Visual C# Express, peut faire beaucoup plus que sa 

fonction basique d'éditeur de fichiers. || possède un outil formidable qui va nous per- 
mettre d'être très efficaces dans le débogage de nos applications et dans la compréhension 
de leur fonctionnement. 


Il s'agit du débogueur. Découvrons vite à quoi il sert et comment il fonctionne | 
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À quoi ça sert ? 


Fidèle à son habitude de nous simplifier la vie, Visual C;# Express possède un débo- 
gueur. C’est un outil très pratique qui va permettre d’obtenir plein d'informations sur 
le déroulement d’un programme. 


Il va permettre d'exécuter les instructions les unes après les autres, de pouvoir observer 
le contenu des variables, de revenir en arrière, bref, de savoir exactement ce qui se passe 
et surtout pourquoi tel bout de code ne fonctionne pas. 


Pour l'utiliser, il faut que Visual C# Express soit en mode « débogage ». Je n’en ai 
pas encore parlé. Pour ce faire, il faut exécuter l’application en appuyant sur (F5) ou 
alors passer par le menu Déboguer > Démarrer le débogage, ou encore cliquer sur le 
petit triangle vert dont l'icône rappelle un bouton de lecture (voir figure 15.1). 


[E] MaPremiereApplication - Microsoft Visual C* 2010 Express (Administrateur) 
Fichier Edition Affichage Refactoriser Projet Générer Déboquer Données Outils 


FIGURE 15.1 - Démarrer le débogage de l’application en cliquant sur le triangle vert 


Pour étudier le débogueur, reprenons la dernière méthode du précédent TP : 


1| static void Main(string[] args) 

2| { 

3 Console.WriteLine (CalculSommelntersection()); 
al} 

5 

6| static int CalculSommelntersection() 

7T|t 

8 List<int> multiplesDe3 = new List<int>(); 
9 List<int> multiplesDe5 = new List<int>(); 
10 

11 for (int i = 1; i <= 100; i++) 

12 { 

13 if (Ci 4 3 == 0) 

14 multiplesDe3.Add(i); 

15 if (i % 5 == 0) 

16 multiplesDe5.Add(i); 

17 } 

18 

19 int somme = O0; 

20 foreach (int m3 in multiplesDe3) 

21 { 

22 foreach (int m5 in multiplesDe5) 

23 { 

24 if (m3 == m5) 


METTRE UN POINT D'ARRÊT ET AVANCER PAS À PAS 


25 somme += m3; 
26 12 

27 } 

28 return somme; 

29 | } 


Mettre un point d’arrêt et avancer pas à pas 


Pour que le programme s’arrête sur un endroit précis et qu’il nous permette de voir ce 
qui se passe, il va falloir mettre des points d’arrêt dans notre code. 


Pour mettre un point d’arrêt, il faut se positionner sur la ligne où nous souhaitons 
nous arrêter, par exemple la première ligne où nous appelons Console.WriteLine et 
appuyer sur (Fo). Nous pouvons également cliquer dans la marge à gauche pour produire 
le même résultat. Un point rouge s’affiche et indique qu’il y a un point d’arrêt à cet 
endroit, ainsi que vous pouvez le voir à la figure 15.2. 


L#| MaPremiereApplication - Microsoft Visual C# 2010 Express (Administrateur) 
Fichier Edition Affichage Refactoriser Projet Générer Déboguer Données Outils Fenêtre 2? 
PSS ISERE u-g-G|+ : 


-| | [| Response 


E 30 Main(stringl] nn) E 


using System.Numerics; 
Elnamespace MaPremiereApplication 


E) class Program 


| { 


E static void Main(string[] args) 


{ 
.WriteLine(CalculSommeIntersection()); 


} 


El static int CalculSommeIntersection() 

{ 
List<int> multiplesDe3 = new List<int>(); 
List<int> multiplesDeS = new List<int>(); 


FIGURE 15.2 — Pose d’un point d’arrêt 


Il est possible de mettre autant de points d’arrêt que nous le souhaitons, à n’importe 
quel endroit de notre code. 


Lançons l’application avec (F5) Nous pouvons voir que Visual C# Express s’arrête sur 
la ligne prévue en la surlignant en jaune (voir figure 15.3). 


Nous en profitons pour remarquer au niveau du carré rouge, que le débogueur est en 
pause et qu’il est possible, soit de continuer l’exécution (triangle), soit de l’arrêter 
(carré), soit enfin de redémarrer le programme depuis le début (carré avec une flèche 
blanche dedans). Nous pouvons désormais exécuter notre code pas à pas, en nous 
servant des icônes à côté ou des raccourcis clavier indiqués à la figure 15.4. 
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L#] MaPremiereApplication (Débogage) - Microsoft Visual C# 2010 Express (Administrateur) 
CE CR LOIFECILE CLEO 
OR +RlEE|= 21099488 Q.1r u aol ac 


— MaPremiereApplication.Program 
using System.Numerics; 


Elnamespace MaPremiereApplication 


{ 


E class Program 


{ 
E static void Main(string[] args) 
{ 
j Fonsole.WriteLine(CaleulSommeïntersection());] 


E static int CalculSommeIntersection() 


List<int> multiplesDe3 = new List<int>(); 


1innetnhe tete teen se 1 Ses an PV 


FIGURE 15.3 — Visual C# Express est arrêté sur une ligne du code 


LE] MaPremiereApplication (Débogage) - Microsoft Visual C£ 2010 Express (Administrateur) 
Fichier Edition Affichage Projet Générer Déboguer Données D Pas à pas détailé (F1) D M Pas à pas sortant (Maj+F11) JS 
MEME IEEE IR u-E-B|»r AU AU 

r[=2100949Q./ruu0|0 0ln 410. 


Pas à pas principal (F10) Ê 


$ MaPremiereApplication.Program 
using System.Numerics; 


Elnamespace MaPremiereApplication 


{ 
E class Program 
{ 
E static void Main(string[] args) 
{ 
i onsole ] 


FIGURE 15.4 - Les boutons permettant d'avancer pas à pas dans l’application 
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OBSERVER DES VARIABLES 


Utilisons la touche pour continuer l’exécution du code pas à pas. La ligne suivante 


se trouve surlignée à son tour. Appuyons à nouveau sur la touche pour aller à 
l'instruction suivante ; il n’y en a plus : le programme se termine. 


Vous me direz : « c’est bien beau, mais nous ne sommes pas passés dans la méthode 
CalculSommelntersection() ». Eh oui, c’est parce que nous avons utilisé la touche 
qui est le pas à pas principal. Pour rentrer dans la méthode, il aurait fallu utiliser 


la touche qui est le pas à pas détaillé. 


Si nous souhaitons que le programme se poursuive pour aller jusqu’au prochain point 
d'arrêt, il suffit d'appuyer sur le triangle (play) ou sur [F5) 


Relançons notre programme en mode débogage, et cette fois-ci, lorsque le débogueur 
s'arrête sur notre point d’arrêt, appuyons sur pour rentrer dans le corps de la mé- 
thode. Voilà, nous sommes dans la méthode CalculSommelntersection(). Continuons 
à appuyer plusieurs fois sur afin de rentrer dans le corps de la boucle for. 


Observer des variables 


À ce moment-là du débogage, si nous passons la souris sur la variable i, qui est l’indice 
de la boucle, Visual C:# Express va nous afficher une mini-information (voir figure 
15.5). 


LIST<INT> MUITIPIESUES = NEW LIST<INT>(); 
List<int> multiplesDeS = new List<int>(); 


for (int i = 1; i <= 100; i++) 


{ oil 
EN 
multiplesDe3.Add(i); 
if (i%5 0) 
multiplesDes.Add(i); 
} 
100% + 4 


FIGURE 15.5 — Visualisation du contenu d’une variable 


Il nous indique que i vaut 1, ce qui est normal, car nous sommes dans la première 
itération de la boucle. Si nous continuons le parcours en appuyant sur plusieurs 
fois, nous voyons que la valeur de i augmente, conformément à ce qui est attendu. 
Maintenant, mettons un point d'arrêt (avec la touche (F9) sur la ligne : 


1 | multiplesDe3.Add(i); 


et poursuivons l’exécution en appuyant sur (F5 |. Il s’arrête au moment où i vaut 3. 


Appuyons sur pour exécuter l’ajout de i à la liste et passons la souris sur la liste 
(voir figure 15.6). 


Visual C# Express nous indique que la liste multiplesDe3 a son Count qui vaut 1. Si 
nous cliquons sur le + pour déplier la liste, nous pouvons voir que 3 a été ajouté dans 
le premier emplacement de la liste (indice 0). Si nous continuons l’exécution plusieurs 
fois, nous voyons que les listes se remplissent au fur et à mesure (voir figure 15.7). 
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if (i% 3 == 0) 
ultiplesDe3.Add(i); 


Æ © multiplesDe3| Count = 1 


multiplesDeS.Add(i); 


FIGURE 15.6 -— Visualisation d’une liste 


for (int i = 1; i <= 100; i++) 


if (i%3 == 0) 


E 4 multiplesDe3 Count = 1 
multiples [0] 3 
} Œ @ Affichage brut 


FIGURE 15.7 - Visualisation d’un élément de la liste 


Enlevez le point d’arrêt sur la ligne en appuyant à nouveau sur et mettez un 
nouveau point d'arrêt sur la ligne : 


1 | int somme = O0; 


Poursuivez l’exécution avec (F5), la boucle est terminée, nous pouvons voir à la figure 
15.8 que les listes sont complètement remplies ! 


if (i%3=—0) 
multiplesDe3.Add(i}; 

if (i % 5 -E multiples De>Count=33 
multiples + } 5 

PAURE 

1]16 

©1219 

foreach (int m3 i 9] 12 èsDe3) 

© [4] 15 


© 151 18 
6 2 


| Valeu o 171 124 


C2 


< 
& 
8 


© [14] 45 


L4 


FIGURE 15.8 -— Visualiser tous les éléments de la liste 


Grâce au débogueur, nous pouvons voir vraiment tout ce qui nous intéresse, c’est une 
des grandes forces du débogueur et c’est un atout vraiment très utile pour comprendre 
ce qui se passe dans un programme (en général, ça se passe mal !). 


Il est également possible de voir les variables locales en regardant en bas dans la fenêtre 
Variables locales. Dans cette fenêtre, vous pourrez observer les variables qui sont 
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couramment accessibles. Il existe également une fenêtre Espion qui permet, de la même 
façon, de surveiller une ou plusieurs variables précises (voir figure 15.9). 


_| Valeur 


| © multiplesDe3 
® muitiplesDes 
@ somme 


51 Variables locales BE EE AT) 


FIGURE 15.9 — La fenêtre des variables locales permet de voir le contenu des variables 
qui sont dans la portée 


Revenir en arrière 


Nom de Zeus, Marty! On peut revenir dans le passé ? 


C’est presque ça, Doc! Si vous souhaitez réexécuter un bout de code, parce que vous 
n'avez pas bien vu ou que vous avez raté l'information qu'il vous fallait, vous pouvez 
forcer le débogueur à revenir en arrière dans le programme. Pour cela, vous avez deux 
solutions. Soit vous faites un clic droit à l’endroit souhaité et vous choisissez l’élément 
de menu Définir l’instruction suivante, comme indiqué à la figure 15.10; soit 
vous déplacez avec la souris la petite flèche jaune sur la gauche qui indique l’endroit 
où nous en sommes, comme l’illustre la figure 15.11. 


Il faut faire attention, car ce retour en arrière n’en est pas complètement un. En effet, 
si vous revenez au début de la boucle qui calcule les multiples et que vous continuez 
l'exécution, vous verrez que la liste continue à se remplir. À la fin de la boucle, au lieu 
de contenir 33 éléments, la liste des multiples de 3 en contiendra 66. En effet, à aucun 
moment nous n’avons vidé la liste et donc le fait de réexécuter cette partie de code 
risque de provoquer des comportements inattendus. Ici, il vaudrait mieux revenir au 
début de la méthode afin que la liste soit de nouveau créée. 


Même si ça ne vous saute pas aux yeux pour l'instant, vous verrez à l’usage que cette 
possibilité est bien pratique. D'autant plus quand vous aurez bien assimilé toutes les 
notions de portée de variable. 


Il est important de noter que, bien que puissant, le débogueur de la version 


gratuite de Visual Studio est vraiment moins puissant que celui des versions 
payantes. || y a un certain nombre de choses que l'on ne peut pas faire. 
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A  Atteindre la définition F12 

Rechercher toutes les références Ctri+K, R 

Point d'arrêt » 
6j Ajouter un espion 


Epingler à la source 


+ Afficher l'instruction suivante Alt+* (pavé numérique) 
oO Pas à pas principal dans les propriétés et les opérateurs 
#5 Exécuter jusqu'au curseur Ctri+F10 
RENE + Définir l'instruction suivante rl+ Maj 
# Couper Ctri+X 
| Nom | 3 Copier Ctri+C 
%, Coller Ctrl+V 


4 multiplesDeS 
9 somme 


Mode Plan 


FIGURE 15.10 — Revenir en arrière dans l’application 


1T (1% > ==) 
multiplesDes.Add(i); 


foreach (int m3 in multiplesDe3) 


FIGURE 15.11 - La flèche jaune indique la future ligne de code à exécuter 


Mais rassurez-vous, celui-ci est quand même déjà bien avancé et permet de faire beau- 
coup de choses. 


La pile des appels 


Une autre fenêtre importante à regarder est la pile des appels. Elle permet d’indiquer 
où nous nous trouvons et par où nous sommes passés pour arriver à cet endroit-là. 


Par exemple, si vous appuyez sur et que vous rentrez dans la méthode 
CalculSommelntersection(), vous pourrez voir dans la pile des appels que vous êtes 
dans la méthode CalculSommelntersection(), qui a été appelée depuis la méthode 
MainO (voir figure 15.12). 


Si vous cliquez sur la ligne du Main(), Visual C# Express vous ramène automatique- 
ment à l’endroit où a été fait l’appel. Cette ligne est alors surlignée en vert pour bien 
faire la différence avec le surlignage en jaune qui est vraiment l’endroit où se trouve le 
débogueur. C’est très pratique quand on à beaucoup de méthodes qui s'appellent les 
unes à la suite des autres (voir figure 15.13). 


La pile des appels est également affichée lorsqu'on rencontre une erreur, elle 
permettra d'identifier plus facilement où est le problème. Vous l'avez vu dans 
le chapitre sur les conversions entre des types incompatibles. 
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Pile des appels 

Nom 

MaPremiereApplication.exe!MaPremiereApplication.Program.CalculSommelntersection() Ligne 29 
MäPremiereApplication.exe!MaPremiereApplication.Program.Main(string[] args) Ligne 13 + 0x5 octets 
[Code externe] 


Fe Pile des appels = RSC OST) 


FIGURE 15.12 - La pile des appels 


li, class Program 
E L static void Main(string[] args) 
("2 . Console.WriteLine(CalculSommeIntersection());] 
} 
E static int CalculSommelntersection() 


Pile des appels 


| Nom 
Ja) _ 3 
©! MaPremiereApplication.exe/MaPremiereApplication.Program.CalculSommelntersection() Ligne 29 


[Code externe] 


É' Pile des appels PRET 


FIGURE 15.13 — Pointer la méthode qui encapsule la ligne courante 
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En tout cas, le débogueur est vraiment un outil à forte valeur ajoutée. Je ne vous ai 
présenté que le strict minimum nécessaire et indispensable. Mais croyez-moi, vous aurez 
l’occasion d’y revenir ! 


En résumé 


— Le débogueur est un outil très puissant permettant d’inspecter le contenu des va- 
riables lors de l’exécution d’un programme. 

— On peut s'arrêter à un endroit de notre application grâce à un point d’arrêt. 

— Le débogueur permet d'exécuter son application pas à pas et de suivre son évolution. 
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TP : le jeu du plus ou du moins 


Difficulté : 1 


aouh , nous augmentons régulièrement le nombre de choses que nous savons. C'est 
super. Nous commençons à être capables d'écrire des applications qui ont un peu 
plus de panache! 


Enfin... moi, j'y arrive! Et vous ? C'est ce que nous allons vérifier avec ce TP. 


Savoir interagir avec son utilisateur est important. Voici donc un petit TP sous forme de 
création d'un jeu simple qui va vous permettre de vous entraîner. L'idée est de réaliser le 
jeu classique du plus ou du moins... 


C# 
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Instructions pour réaliser le TP 


Je vous rappelle les règles. L'ordinateur calcule un nombre aléatoire et nous devons le 
deviner. À chaque saisie, il nous indique si le nombre saisi est plus grand ou plus petit 
que le nombre à trouver. Une fois trouvé, il nous indique en combien de coups nous 
avons réussi à trouver le nombre secret. 


Pour ce TP, vous savez presque tout faire. Il ne vous manque que l'instruction pour 
obtenir un nombre aléatoire. La voici, cette instruction permet de renvoyer un nombre 
compris entre 0 et 100 (exclu). Ne vous attardez pas trop sur sa syntaxe, nous aurons 
l’occasion de comprendre exactement de quoi il s’agit dans la partie suivante : 


| int valeurATrouver = new Random().Next (0, 100); 


Le principe est grosso modo le suivant : tant qu’on n’a pas trouvé la bonne valeur, nous 
devons en saisir une nouvelle. Dans ce cas, la console nous indique si la valeur est trop 
grande ou trop petite. Il faudra bien sûr incrémenter un compteur de coups à chaque 
essai. 


N'oubliez pas de gérer le cas où l'utilisateur saisit n’importe quoi. Nous ne voudrions 
pas que notre premier jeu ait un bug qui fasse planter l’application ! 


Allez, je vous en ai trop dit. C’est à vous de jouer. Bon courage. 
Correction 
Voici ma correction de ce TP. 


Bien sûr, il existe beaucoup de façons de réaliser ce petit jeu. S'il fonctionne, c’est que 
votre solution est bonne. Ma solution fonctionne, la voici : 


1| static void Main(string[] args) 

2| { 

3 int valeurATrouver = new Random().Next (O0, 100); 

4 int nombreDeCoups = 0; 

5 bool trouve = false; 

6 Console.WriteLine("Veuillez saisir un nombre compris entre 
0 et 100 Cexclu)"); 

7 while (!trouve) 

8 { 
string saisie = Console.ReadLine(); 

10 int valeurSaisie; 

11 if (int.TryParse(saisie, out valeurSaisie)) 

12 { 

13 if (valeurSaisie == valeurATrouver) 

14 trouve = true; 

15 else 

16 € 

17 if (valeurSaisie < valeurATrouver) 

18 Console.WriteLine ("Trop petit ..."); 
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19 else 

20 Console.WriteLine("Trop grand ..."); 

21 } 

22 nombreDeCoups++; 

23 } 

24 else 

25 Console.WriteLine("La valeur saisie est incorrecte, 

veuillez recommencer ..."); 

26 x 

27 Console.WriteLine ("Vous avez trouvé en " + nombreDeCoups + 
t goup(spe): 

28| } 


On commence par obtenir un nombre aléatoire avec l’instruction que j'ai fournie dans 
l'énoncé. Nous avons ensuite les initialisations de variables. L’entier nombreDeCoups va 
permettre de stocker le nombre d’essai et le booléen trouve va permettre d’avoir une 
condition de sortie de boucle. 


Notre boucle démarre et ne se terminera qu’une fois que le booléen trouve sera passé 
à vrai (true). Dans le corps de la boucle, nous demandons à l’utilisateur de saisir une 
valeur que nous essayons de convertir en entier. Si la conversion échoue, nous l’indiquons 
à l’utilisateur et nous recommençons notre boucle. Notez ici que je n’incrémente pas le 
nombre de coups, jugeant qu’il n’y a pas lieu de pénaliser le joueur parce qu’il a mal 
saisi ou qu'il a renversé quelque chose sur son clavier juste avant de valider la saisie. 


Si en revanche, la conversion se passe bien, nous pouvons commencer à comparer la 
valeur saisie avec la valeur à trouver. Si la valeur est la bonne, nous passons le booléen 
à vrai, ce qui nous permettra de sortir de la boucle et de passer à la suite. Sinon, 
nous afficherons un message pour indiquer si la saisie est trop grande ou trop petite 
en fonction du résultat de la comparaison. Dans tous les cas, nous incrémenterons le 
nombre de coups. Enfin, en sortie de boucle, nous indiquerons sa victoire au joueur 
ainsi que le nombre de coups utilisés pour trouver le nombre secret : 


Veuillez saisir un nombre compris entre O0 et 100 (exclu) 
50 
Propapertbre 

petit. 

grand.. 


Pet bee 


grand.. 


avez trouvé en 6 coup(s) 
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Aller plus loin 


Il est bien sûr toujours possible d'améliorer le jeu. Nous pourrions par exemple ajouter 
un contrôle sur les bornes de la saisie. Ainsi, si l'utilisateur saisit un nombre supérieur ou 
égal à 100 ou inférieur à 0, nous pourrions lui rappeler les bornes du nombre aléatoire. 


De même, à la fin, plutôt que d’afficher « coup(s) », nous pourrions tester la valeur du 
nombre de coups. S’il est égal à 1, on affiche « coup » au singulier, sinon « coups » au 
pluriel. 


La boucle pourrait également être légèrement différente. Plutôt que de tester la condi- 
tion de sortie sur un booléen, nous pourrions utiliser le mot-clé break. De même, nous 
pourrions alléger l'écriture avec le mot-clé continue. Par exemple : 


1| static void Main(string[] args) 

21 { 

3 int valeurATrouver = new Random().Next (O0, 100); 

4 int nombreDeCoups = 0; 

5 Console.WriteLine("Veuillez saisir un nombre compris entre 
O et 100 (exclu)") : 

6 while (true) 

7 { 

8 string saisie = Console.ReadLine() ; 

9 int valeurSaisie; 

10 if (lint.TryParse(saisie, out valeurSaisie)) 

11 { 

12 Console.WriteLine("La valeur saisie est incorrecte, 

veuillez recommencer ..."); 

13 continue; 

14 } 

15 if (valeurSaisie < 0 || valeurSaisie >= 100) 

16 € 

17 Console.WriteLine("Vous devez saisir un nombre 

éntre 0 et 100 exclu :.:M): 

18 continue; 

19 } 

20 nombreDeCoups++; 

21 if (valeurSaisie == valeurATrouver) 

22 break; 

23 if (valeurSaisie < valeurATrouver) 

24 Console.WriteLine ("Trop petit ..."); 

25 else 

26 Console.WriteLine("Trop grand ..."); 

27 } 

28 if (nombreDeCoups == 1) 

29 Console.WriteLine ("Vous avez trouvé en " + 

nombreDeCoups + " coup"); 
30 else 
31 Console.WriteLine ("Vous avez trouvé en " + 
nombreDeCoups + " coups"); 
32| } 


ALLER PLUS LOIN 


Tout ceci est une question de goût. Je préfère personnellement la version précédente 
n’aimant pas trop les break et les continue. Mais après tout, chacun fait comme il 
préfère, l’important est que nous nous amusions à écrire le programme et à y jouer! 


Voilà, ce TP est terminé. Vous avez pu voir finalement que nous étions tout à fait 
capables de réaliser de petites applications récréatives. Personnellement, jai commencé 
à m’amuser à faire de la programmation en réalisant toute sorte de petits programmes 
de ce genre. 


Je vous encourage fortement à essayer de créer d’autres programmes par vous-mêmes. 
Plus vous vous entraînerez à faire des petits programmes simples et plus vous réussirez 
à appréhender les subtilités de ce que nous avons appris. 


Bien sûr, plus tard, nous serons capables de réaliser des applications plus compliquées. .…. 
Cela vous tente ? Alors continuons la lecture! 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code ) 


Code web : 694204 
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OR RE 


La ligne de commande 


Difficulté : 16 


ous ce nom un peu barbare se cache une fonctionnalité très présente dans nos usages 

quotidiens mais qui a tendance à être masquée à l'utilisateur lambda. Nous allons, 

dans un premier temps, voir ce qu'est exactement la ligne de commande et à quoi elle 
sert. Ensuite, nous verrons comment l'exploiter dans notre application avec le C#. 


Notez que ce chapitre n'est pas essentiel mais qu'il pourra sûrement vous servir plus tard 
dans la création de vos applications. 
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Qu'est-ce que la ligne de commande ? 


La ligne de commande, c’est ce qui nous permet d’exécuter nos programmes. Très 
présente à l’époque où Windows existait peu, elle a tendance à disparaître de nos 
utilisations. Mais pas le fond de son fonctionnement. 


En général, la ligne de commande sert à passer des arguments à un programme. Par 
exemple, pour ouvrir un fichier texte avec le bloc-notes (notepad.exe), on peut le faire 
de deux façons différentes. Soit on ouvre le bloc-notes et on va dans le menu Fichier > 
Ouvrir puis on va chercher le fichier pour l’ouvrir. Soit on utilise la ligne de commande 
pour l’ouvrir directement. 


Pour ce faire, il suffit d’utiliser la commande : notepad c:\test.txt et le fichier 
c:\test.txt s'ouvre directement dans le bloc-notes. 


Ce qu'il s’est passé derrière, c’est que nous avons demandé d’exécuter le programme 
notepad.exe avec le paramètre c:\test.txt. Le programme notepad a analysé sa 
ligne de commande, il y a trouvé un paramètre et il a ouvert le fichier correspondant. 


Superbe! Nous allons apprendre à faire pareil! 


Passer des paramètres en ligne de commande 


La première chose est de savoir comment faire pour passer des paramètres en ligne de 
commande. Plusieurs solutions existent. On peut, par exemple, exécuter la commande 
que j'ai écrite plus haut depuis le menu Démarrer > Exécuter ou la combinaison touche 


Windows + (R) (voir la figure 17.1). 


eee (mem Sn) 


ps Entrez le nom d'un programme, dossier, document ou ressource Internet, et 
Windows l'ouvrira pour vous. 


Ouvrir: | notepad c:\test.bdl + 


æ Cette tâche sera créée avec les autorisations d'administrateur. 


[_x_ ]| Annuler | | Parcourir... | 


FIGURE 17.1 - Exécuter une application avec la ligne de commande 


On peut également le faire depuis une invite de commande, en allant dans : Menu 
Démarrer > Accessoires > Invite de commande. 


Ceci nous ouvre une console noire, comme celle que l’on connaît bien et dans laquelle 
nous pouvons taper des instructions (voir figure 17.2). 


Mais nous, ce qui nous intéresse surtout, c’est de pouvoir le faire depuis Visual C# 
Express afin de pouvoir passer des arguments à notre programme. Pour ce faire, on va 
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LE Aûmi EI) 
Administrateur : Invite de commandes — 


Microsoft Windows [uersion 6.1.76611] 
Copyright <c)> 2889 Microsoft Corporation. Tous droits réservés. 


GC: \Users\Nicodnotepad c:\test.txt 


FIGURE 17.2 — Exécuter une application depuis l’invite de commande 


aller dans les propriétés de notre projet (clic droit sur le projet, Propriétés) puis nous 
allons dans l’onglet Déboguer et nous voyons une zone de texte permettant de mettre 
des arguments à la ligne de commande. Rajoutons par exemple « Bonjour Nico » (voir 


figure 17.3). 


 _ 


Application 


Options de démarrage 


Géné - 
a Arguments de la ligne de commande: | Bonjour Nico 


Événements de build 


| Débogue 


Ressources Répertoire de travail : 


Paramètres Y| Activer le processus d'hébergement Visual Studio 


Chemins d'accès des références 
Signature 


Sécurité 


«! 


FIGURE 17.3 —- Modifier les arguments de la ligne de commande 


Voilà, maintenant lorsque nous exécuterons notre application, Visual C# Express lui 
passera les arguments que nous avons définis en paramètre de la ligne de commande. 
À noter que dans ce cas, les arguments Bonjour Nico ne seront valables que lorsque 
nous exécuterons l'application à travers Visual C#£ Express. Évidemment, si nous exé- 
cutons notre application par l'invite de commande, nous aurons besoin de repasser les 
arguments au programme pour qu'il puisse les exploiter. 
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C’est bien! Sauf que pour l'instant, ça ne nous change pas la vie! Il faut apprendre à 
traiter ces paramètres... 


Lire la ligne de commande 


On peut lire le contenu de la ligne de commande de deux façons différentes. Vous 
vous rappelez de la méthode Main () ? On a vu que Visual C# Express générait cette 
méthode avec des paramètres. Eh bien ces paramètres, vous ne devinerez jamais! Ce 
sont les paramètres de la ligne de commande. Oh joie! 


1| static void Main(string[] args) 
2| { 
3| } 


La variable args est un tableau de chaînes de caractères. Sachant que chaque paramètre 
est délimité par des espaces, nous retrouverons chacun des paramètres à un indice du 
tableau différent. 


Ce qui fait que si j'utilise le code suivant : 


1| foreach (string parametre in args) 
2| € 

3 Console .WriteLine (parametre); 
al} 


et que je lance mon application avec les paramètres définis précédemment, je vais 
obtenir : 


Bonjour 
Nico 


Et voilà, nous avons récupéré les paramètres de la ligne de commande, il ne nous 
restera plus qu’à les traiter dans notre programme. Comme prévu, nous obtenons deux 
paramètres. Le premier est la chaîne de caractères « Bonjour », le deuxième est la 
chaîne de caractères « Nico ». N'oubliez pas que c’est le caractère d’espacement qui 
sert de délimiteur entre les paramètres. 


L'autre façon de récupérer la ligne de commande est d'utiliser la méthode 
Environment .GetCommandLineArgs(). Elle renvoie un tableau contenant les para- 
mètres, comme ce qui est passé en paramètres à la méthode Main. La seule différence, 
c’est que dans le premier élément du tableau, nous trouverons le chemin complet de 
notre programme. Ceci peut être utile dans certains cas. Si cela n’a aucun intérêt pour 
vous, il suffira de commencer la lecture à partir de l’indice numéro 1. 


L'exemple suivant, affiche toutes les valeurs du tableau : 


1| foreach (string parametre in Environment .GetCommandLineArgs()) 
2| € 

3 Console .WriteLine (parametre); 

al} 
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Ce qui donne : 


C:\Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\bin\Release\ 


MaPremiereApplication.exe 
Bonjour 
Nico 


Attention : si vous devez accéder à un indice précis du tableau, vérifiez bien 
que la taille du tableau le permet. N'oubliez pas que si le tableau contient 
un seul élément et que vous essayez d'accéder au deuxième élément, alors il 
ÿ aura une erreur. 


Tu as dit que c'était le caractère d'espacement qui permettait de déli- 
miter les paramètres. J'ai essayé de rajouter le paramètre C:\Program 
Files\test.txt à ma ligne de commande afin de changer l'emplacement 
du fichier, mais je me retrouve avec deux paramètres au lieu d'un seul, c'est 
normal ? 


Eh oui, il y a un espace entre Program et Files. L’astuce est de passer le paramètre 
entre guillemets, de cette façon : "C:\Program Files\test.txt". Nous aurons un seul 
paramètre dans la ligne de commande à la place de deux. Comme par exemple à la 
figure 17.4. 


APP Options de démarrage 
ME Arguments de la ligne de commande: ["mon premier paramètre""mon | 
Événements de build deuxième paramètre" 

| Déboguer + 
Ressources Répertoire de travail : (ea) 
Paramètres I] Activer le processus d'hébergement Visual Studio 


FIGURE 17.4 - Modifier les arguments de la ligne de commande 


En résumé 
— On peut passer des paramètres à une application en utilisant la ligne de commande. 


— Le programme C# peut lire et interpréter ces paramètres. 
— Ces paramètres sont séparés par des espaces. 
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sie LB | 


TP : calculs en ligne de commande 


Difficulté : 16 


t voici un nouveau TP qui va nous permettre de récapituler un peu tout ce que nous 
avons vu. Au programme, des if, des switch, des méthodes, des boucles et... de la 
ligne de commande bien sûr. 


Grâce à nos connaissances grandissantes, nous arrivons à augmenter la difficulté de nos 
exercices et c'est une bonne chose. Vous verrez que dans le moindre petit programme, vous 
aurez besoin de toutes les notions que nous avons apprises. 


Allez, ne traînons pas trop et à vous de travailler! 


C# 
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Instructions pour réaliser le TP 


L'objectif de ce TP est de réaliser un petit programme qui permet d'effectuer une 
opération mathématique à partir de la ligne de commande. 


Il va donc falloir écrire un programme que l’on pourra appeler avec des paramètres. En 
imaginant que le programme s'appelle MonProgramme, on pourrait l’utiliser ainsi pour 
faire la somme de deux nombres : 


— MonProgramme somme 2 5 

Ce qui renverra bien sûr 7. 

Pour effectuer une multiplication, nous pourrons faire : 
— MonProgramme multiplication 2 5 

et nous obtiendrons 10. 

Enfin, pour calculer une moyenne, nous pourrons faire : 
— MonProgramme moyenne 1 2345 

ce qui renverra 3. 

Les règles sont les suivantes : 


— il doit y avoir deux et seulement deux nombres composant l’addition et la multipli- 
cation ; 

— la moyenne peut contenir autant de nombres que souhaité ; 

— si le nombre de paramètres est incorrect, alors le programme affichera un message 
d’aide explicite ; 

— si le nombre attendu n’est pas un double correct, on affichera le message d’aide ; 


Plutôt simple, non? Alors, à vous de jouer, il n’y a pas de subtilité. 
Correction 
STOP ! Je relève les copies. 


J’ai dit qu’il n’y avait pas de subtilité, mais j’ai un peu menti en fait. Je l’avoue, c'était 
pour que vous alliez au bout du développement sans aide. 


Pour vérifier que vous avez trouvé la subtilité, essayez votre programme avec les para- 
mètres suivants : 


MonProgramme addition 5,5 2,5 
Q Obtenez-vous 8 ? 


C'était à peu près la seule subtilité, il fallait juste savoir qu’un nombre à virgule s’écri- 
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CORRECTION 


vait avec une virgule plutôt qu'avec un point. Si nous écrivons 2.5, alors le C# ne sait 
pas faire la conversion. 


En vrai, c'est un peu plus compliqué que ça et nous le verrons dans un 
prochain chapitre, mais l'écriture du nombre à virgule est dépendante de ce 
qu'on appelle la culture courante que l'on peut modifier dans les paramètres 
Horloge, langue et région du panneau de configuration de Windows. Si 
l'on change le « format de la date, de l'heure ou des nombres » on change 
la culture courante, et la façon dont les nombres à virgule (et autres) sont 
gérés est différente. 


Quoi qu’il en soit, voici ma correction du TP : 


1| static void Main(string[] args) 

2| € 

3 if (args.Length == 0) 

4 { 

5 AfficherAide(); 

6 } 

7 else 

8 À 

9 string operateur = args[O]; 

10 switch (operateur) 

11 { 

12 case "addition": 

13 Addition(args); 

14 break; 

15 case "multiplication": 

16 Multiplication(args); 

17 break; 

18 case "moyenne": 

19 Moyenne (args) ; 

20 break; 

21 default: 

22 AfficherAide(); 

23 break; 

24 } 

25 } 

26| } 

27 

28| static void AfficherAide() 

29 | À 

30 Console.WriteLine ("Utilisez l'application de la manière 
éiivante €): 

31 Console.WriteLine("MonProgamme addition 2 5"); 

32 Console.WriteLine('"MonProgamme multiplication 2 5"); 

33 Console.WriteLine ("MonProgamme moyenne 2 5 10 11"); 

34| } 

35 
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36| static void Addition(string[] args) 


37 | À 

38 if (args.Length != 3) 

39 { 

40 AfficherAide); 

41 } 

42 else 

43 Ÿ 

44 double somme = 0; 

45 for (int i = 1; i < args.Length; i++) 

46 { 

47 double valeur; 

48 if (!double.TryParse(argsli], out valeur)) 

49 { 

50 AfficherAide(); 

51 return; 

52 } 

63 somme += valeur; 

54 } 

55 Console.WriteLine("Résultat de l'addition : " + somme); 

56 de 

57 | } 

58 

59| static void Multiplication(string[] args) 

60 | { 

61 if (args.Length != 3) 

62 { 

63 AfficherAide(); 

64 } 

65 else 

66 { 

67 double resultat = 1; 

68 for (int i = 1; i < args.Length; i++) 

69 { 

70 double valeur; 

71 if (ldouble.TryParse(argsli]l, out valeur)) 

72 { 

73 AfficherAide(); 

74 return; 

75 } 

76 resultat *= valeur; 

77 } 

78 Console.WriteLine("Résultat de la multiplication : " + 
resultat); 

79 F 

80| } 

81 

82| static void Moyenne(string[]l args) 

83| { 

84 double total = 0; 


ALLER PLUS LOIN 


85 for (int i = 1; i < args.Length; i++) 

86 { 

87 double valeur; 

88 if (!double.TryParse(argslil, out valeur)) 
89 { 

90 AfficherAide () ; 

91 return; 

92 } 

93 total += valeur; 

94 F 

95 total = total / (args.Length - 1); 

96 Console.WriteLine("Résultat de la moyenne : " + total); 
97 | } 


Disons que c’est plus long que difficile... 


Oui, je sais, lors de l’addition et de la multiplication, j'ai parcouru l’ensemble des 
paramètres alors que j'aurais pu utiliser uniquement args [1] et args[2]. L'avantage, 
c’est que si je supprime la condition sur le nombre de paramètres, alors mon addition 
pourra fonctionner avec autant de nombres que je le souhaite... Vous aurez remarqué 
que je commence mon parcours à l’indice 1, d’où la pertinence de l’utilisation de la 
boucle for plutôt que de la boucle foreach. 


Notez quand même l’utilisation du mot-clé return afin de sortir prématurément de la 
méthode en cas de problème. 


Évidemment, il y a plein de façons de réaliser ce TP, la mienne en est une, il y en a 
d’autres. 


J'espère que vous aurez fait attention au calcul de la moyenne. Nous retirons 1 à la 
longueur du tableau pour obtenir le nombre de paramètres. Il faut donc faire attention 
à l’ordre des parenthèses afin que C# fasse d’abord la soustraction avant la division, 
alors que sans ça, la division est prioritaire sur la soustraction. Sinon, c’est l’erreur de 
calcul assurée, indigne de notre superbe application ! 


Aller plus loin 


Ce code est très bien. Si, si, c’est moi qui l’ai fait! 
Sauf qu’il y à quand même un petit problème... d'ordre architectural. 


Il a été plutôt simple à écrire en suivant l’énoncé du TP mais on peut faire encore 
mieux. On peut le rendre plus facilement maintenable en l’organisant différemment. 


Observez le code suivant : 


1| static void Main(string[] args) 

2| € 

3 bool ok = false; 

4 if (args.Length > 0) 

5 { 

6 string operateur = args[O]; 
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26 


52 


152 


switch (operateur) 
{ 
case "addition": 
ok = Addition(args); 
break; 
case "multiplication": 
ok = Multiplication(args); 
break; 
case "moyenne": 
ok = Moyenne (args); 
break; 


if (lok) 
AfficherAide); 
} 


static void AfficherAide() 
{ 
Console.WriteLine ("Utilisez l'application de la manière 
suivante :"); 
Console.WriteLine("MonProgamme addition 2 5"); 
Console.WriteLine("MonProgamme multiplication 2 5"); 
Console.WriteLine ("MonProgamme moyenne 2 5 10 11"); 


+ 


static bool Addition(string[] args) 
€ 
if (args.Length != 3) 
return false; 


double somme = 0; 
for (int i = 1; i < args.Length; i++) 
{ 
double valeur; 
if (l!double.TryParse(argslil, out valeur)) 
return false; 
somme += valeur; 


} 


Console.WriteLine("Résultat de l'addition : " + somme); 
return true; 


static bool Multiplication(string[] args) 
{ 
if (args.Length != 3) 
return false; 
double resultat = 1; 


ALLER PLUS LOIN 


67 


for (int i = 1; i < args.Length; i++) 
{ 
double valeur; 
if (!double.TryParse(argsli], out valeur)) 
return false; 
resultat *= valeur; 


} 
Console.WriteLine ("Résultat de la multiplication + 
resultat); 
return true; 
} 
static bool Moyenne(string[] args) 
{ 
double total = 0; 
for (int i = 1; i < args.Length; i++) 
{ 
double valeur; 
if (!double.TryParse(argsli], out valeur)) 
return false; 
total += valeur; 
} 
total = total / (args.Length - 1); 
Console.WriteLine ("Résultat de la moyenne : " + total); 
return true; 
} 


Nous n’affichons l’aide qu’à un seul endroit en utilisant le fait que les méthodes renvoient 
un booléen indiquant si l’opération a été possible ou pas. 


D'une manière générale, il est important d’essayer de rendre son code le plus mainte- 
nable possible, comme nous l’avons fait ici. 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 


vant : 
Copier ce code 
Code web : 846482 
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Troisième partie 


Le C#Æ, un langage orienté objet 
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be 1 9 


Introduction à la programmation 
orientée objet 


Difficulté : 


ans ce chapitre, nous allons essayer de décrire ce qu'est la programmation orientée 
objet (abrégée souvent en POO). Je dis bien « essayer », car pour être complètement 
traitée, la POO nécessiterait qu'on lui dédie un ouvrage entier | 


Nous allons tâcher d'aller à l'essentiel et de rester proche de la pratique. Ce n'est pas très 
grave si tous les concepts abstraits ne sont pas parfaitement appréhendés : il est impos- 
sible d'apprendre la POO en deux heures. Cela nécessite une longue pratique, beaucoup 
d'empirisme et des approfondissements théoriques. 


Ce qui est important ici, c'est de comprendre les notions de base et de pouvoir les utili- 
ser dans de petits programmes. Après cette mise au point, attaquons sans plus tarder la 


programmation orientée objet | 
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CHAPITRE 19. INTRODUCTION À LA PROGRAMMATION ORIENTÉE 
OBJET 


Qu'est-ce qu’un objet ? 


Vous avez pu voir précédemment que j’ai utilisé de temps en temps le mot « objet » et 
que le mot-clé new est apparu comme par magie... Il a été difficile de ne pas trop en 
parler et il est temps d’en savoir un peu plus. 


© Alors, qu'est-ce qu'un objet ? 


Si on prend le monde réel (si, si, vous allez voir, vous connaissez. ..), nous sommes 
entourés d'objets : une chaise, une table, une voiture, etc. Ces objets forment un tout. 


— Ils possèdent des propriétés (la chaise possède quatre pieds, elle est de couleur bleue, 
etc.). 

— Ces objets peuvent faire des actions (la voiture peut rouler, klaxonner, etc.). 

— Ils peuvent également interagir entre eux (l’objet conducteur démarre la voiture, 
l’objet voiture fait tourner l’objet volant, etc.). 


Il faut bien faire attention à distinguer ce qu'est l’objet et ce qu'est la définition d’un 
objet. 


La définition de l’objet (ou structure de l’objet) permet d’indiquer ce qui compose un 
objet, c’est-à-dire quelles sont ses propriétés, ses actions etc. Comme par exemple, le 
fait qu’une chaise ait des pieds ou qu’on puisse s'asseoir dessus. Par contre, l’objet 
chaise est bien concret. On peut donc avoir plusieurs objets chaise : on parle également 
d’instances. Les objets chaise, ce sont bien ceux, concrets, que l’on voit devant nous 
autour de l’objet table, pour démarrer une partie de belote. On peut faire l’analogie 
avec notre dictionnaire qui nous décrit ce qu'est une chaise. Le dictionnaire décrit 
en quoi consiste l’objet, et l'instance de l’objet représente le concret associé à cette 
définition. Chaque objet a sa propre vie et diffère d’un autre. Nous pouvons avoir une 
chaise bleue, une autre rouge, une autre avec des roulettes, une cassée... 


Vous voyez, finalement, la notion d’objet est plutôt simple quand on la ramène à ce 
qu'on connaît déjà ! 

Sachant qu’un objet en programmation, c’est comme un objet du monde réel, mais ce 
n’est pas forcément restreint au matériel. Un chien est un objet. Des concepts comme 
l’amour ou une idée sont également des objets, tandis qu’on ne dirait pas cela dans le 
monde réel. 


En conclusion : 


— La définition (ou structure) d’un objet est un concept abstrait, comme une définition 
dans le dictionnaire. Cette définition décrit les caractéristiques d’un objet (la chaise 
a des pieds, l’homme a des jambes, etc.). Cette définition est unique, comme une 
définition dans le dictionnaire. 

— Un objet ou une instance est la réalisation concrète de la structure de l’objet. On 
peut avoir de multiples instances, comme les cent voitures sur le parking devant 
chez moi. Elles peuvent avoir des caractéristiques différentes (une voiture bleue, une 
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voiture électrique, une voiture à cinq portes, etc.). 


L’encapsulation 


Le fait de concevoir une application comme un système d’objets interagissant entre eux 
apporte une certaine souplesse et une forte abstraction. 


Prenons un exemple tout simple : la machine à café du bureau. Nous insérons nos pièces 
dans le monnayeur, choisissons la boisson et nous nous retrouvons avec un gobelet de 
la boisson commandée. Nous nous moquons complètement de savoir comment cela 
fonctionne à l’intérieur et nous pouvons complètement ignorer si le café est en poudre, 
en grain, comment l’eau est ajoutée, chauffée, comment le sucre est distribué, etc. Tout 
ce qui nous importe, c’est que le fait de mettre des sous dans la machine nous donne 
un café qui va nous permettre d’attaquer la journée. 


Voilà un bel exemple de programmation orientée objet. Nous manipulons un objet 
MachineACafe qui a des propriétés (allumée/éteinte, présence de café, présence de 
gobelet,...) et qui sait faire des actions (AccepterMonnaie, DonnerCafe,...). Et c’est 
tout ce que nous avons besoin de savoir : on se moque du fonctionnement interne, peu 
importe ce qui se passe à l’intérieur, notre objet nous donne du café, point! 


C’est ce qu’on appelle l’encapsulation. Cela permet de protéger l'information conte- 
nue dans notre objet et de le rendre manipulable uniquement par ses actions ou pro- 
priétés. Ainsi, l'utilisateur ne peut accéder ni au café ni au sucre et encore moins à la 
monnaie. Notre objet est ainsi protégé et fonctionne un peu comme une boîte noire. 


L'intérêt est que si la personne qui entretient la machine met du café en grain à la 
place du café soluble, c’est invisible pour l'utilisateur qui n’a juste qu’à se soucier de 
mettre ses pièces dans la machine. 


L’encapsulation protège donc les données de l’objet et son fonctionnement interne. 


Héritage 


Un autre élément important dans la programmation orientée objet, que nous allons 
aborder maintenant, est l'héritage. 


Q Ah bon ? Les objets aussi peuvent mourir et transmettre leur patrimoine ? 


Eh bien c’est presque comme en droit, à part que l’objet ne meurt pas et qu’il n’y a pas 
de taxe sur l’héritage. C'est-à-dire qu’un objet dit « père » peut transmettre certaines 
de ses caractéristiques à un autre objet dit « fils ». 


Pour cela, on pourra définir une relation d’héritage entre eux. 


S'il y à une relation d’héritage entre un objet père et un objet fils, alors l’objet fils 
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hérite de l’objet père. On dit également que l’objet fils est une spécialisation de 
l’objet père ou qu’il dérive de l’objet père. 


En langage plus courant, on peut également dire que l’objet fils est « une sorte » d'objet 
père. 


Q Je ne suis pas sûr de comprendre, peux-tu donner des exemples ? 


Alors, prenons par exemple l’objet chien et imaginons ses caractéristiques tirées du 
monde réel en utilisant l'héritage : 


— l’objet chien est une sorte d’objet mammifere ; 
— l’objet mammifere est une sorte d’objet animal ; 
— l’objet animal est une sorte d'objet etreVivant. 


Chaque père est un peu plus général que son fils. Et inversement, chaque fils est un 
peu plus spécialisé que son père. Avec l'exemple du dessus, un mammifère est un peu 
plus général qu’un chien, l’être vivant étant encore plus général qu’un mammifère. 


Il est possible pour un père d’avoir plusieurs fils, par contre, l’inverse est impossible, 
un fils ne peut pas avoir plusieurs pères. Eh oui, c’est triste mais c’est comme ça, c’est 
le règne du père célibataire avec plusieurs enfants à charge! 


Aïnsi, un objet chat peut également être un fils de l’objet mammifere. Un objet vegetal 
peut également être fils de l’objet etreVivant. 


Ce qu’on peut reproduire sur le schéma suivant (voir figure 19.1) où chaque bulle 
représente un objet et chaque flèche représente l’héritage entre les objets. 


Pé Fi 


FIGURE 19.1 - L'héritage entre les objets 
On peut définir une sorte de hiérarchie entre les objets, un peu comme on le ferait avec 
un arbre généalogique. La différence est qu’un objet héritant d’un autre peut obtenir 


certains ou tous les comportements de l’objet qu’il spécialise, alors qu’un enfant n’hérite 
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pas forcément des yeux bleus de sa mère ou du côté bougon de son grand-père, le hasard 
de la nature faisant le reste. 


Pour bien comprendre cet héritage de comportement, empruntons à nouveau les exemples 
au monde réel. 


— L'être vivant peut, par exemple, faire l’action « vivre ». 

— Le mammifère possède des yeux. 

— Le chien, qui est une sorte d’être vivant et une sorte de mammifère, peut également 
faire l’action « vivre » et aura des yeux. 

— Le chat, qui est une autre sorte d’être vivant, peut lui aussi faire l’action « vivre » 
et aura également des yeux. 


On voit bien ici que le chat et le chien héritent des comportements de leurs parents et 
grands-parents en étant capables de vivre et d’avoir des yeux. 


Par contre, l’action « aboyer » est spécifique au chien. Ce qui veut dire que ni le chat 
ni le dauphin ne seront capables d’aboyer. Il n’y a que dans les dessins animés de Tex 
Avery que ceci est possible! Évidemment, il n’y a pas de notion d’héritage entre le 
chien et le chat et l’action d’aboyer est définie au niveau du comportement du chien. 


Ceci implique également que seul un objet qui est une sorte de chien, par exemple 
l’objet labrador ou l’objet chihuahua, pourra hériter du comportement « aboyer », car 
il y a une relation d’héritage entre eux. 


Finalement, c’est plutôt logique ! 


Rappelons juste avant de terminer ce paragraphe qu’un objet ne peut pas hériter de plu- 
sieurs objets. Il ne peut hériter que d’un seul objet. Le C# ne permet pas ce qu’on ap- 
pelle l’héritage multiple, a contrario d’autres langages comme le C++ par exemple. 


Voilà globalement pour la notion d’héritage. 


Je dis globalement, car il y a certaines subtilités que je n’ai pas abordées mais ce 
n’est pas trop grave; vous verrez dans les chapitres suivants comment le C# utilise la 
notion d’héritage et ce qu’il y à vraiment besoin de savoir. Ne vous inquiétez pas si 
certaines notions sont encore un peu floues, vous comprendrez sûrement mieux grâce à 
la pratique. 


Polymorphisme - Substitution 


Le mot polymorphisme suggère qu’une chose peut prendre plusieurs formes. Sous ce 
terme un peu barbare se cachent plusieurs notions de l’orienté objet qui sont souvent 
source d’erreurs. 


Je vais volontairement passer rapidement sur certains points qui ne vont pas nous servir 
pour me concentrer sur ceux qui sont importants pour ce livre. 


En fait, on peut dire qu’une manifestation du polymorphisme est la capacité pour un 
objet de faire une même action avec différents types d'intervenants. C’est ce qu’on 
appelle le polymorphisme ad hoc ou le polymorphisme « paramétré ». Par exemple, 
notre objet voiture peut rouler sur la route, rouler sur l’autoroute, rouler sur la terre 
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si elle est équipée de pneus adéquats, rouler au fond de l’eau si elle est amphibie, etc. 


Concrètement ici, je fais interagir un objet voiture avec un objet autoroute ou un 
objet terre... par l’action qui consiste à rouler. Cela peut paraître anodin décrit 
ainsi, mais nous verrons ce que cela implique avec le C#. 


La substitution est une autre manifestation du polymorphisme. Il s’agit de la capacité 
d’un objet fils à redéfinir des caractéristiques ou des actions d’un objet père. 


Prenons par exemple un objet mammifère qui sait faire l’action « se déplacer ». Les 
objets qui dérivent du mammifère peuvent potentiellement avoir à se déplacer d’une 
manière différente. Par exemple, l’objet homme va se déplacer sur ses deux jambes et 
donc différemment de l’objet dauphin qui se déplacera grâce à ses nageoires ou bien 
encore différemment de l’objet « homme accidenté » qui va avoir besoin de béquilles 
pour s’aider dans son déplacement. 


Tous ces mammifères sont capables de se déplacer, mais chacun va le faire d’une manière 
différente. Ceci est donc possible grâce à la substitution qui permet de redéfinir un 
comportement hérité. 


Ainsi, chaque fils sera libre de réécrire son propre comportement, si celui de son père 
ne lui convient pas. 


Interfaces 


Un autre concept important de la programmation orientée objet est la notion d’inter- 
face. 


général un comportement. 


(d) L'interface est un contrat que s'engage à respecter un objet. Il indique en 


Prenons un exemple dans notre monde réel et connu : les prises de courant. 


Elles fournissent de l'électricité à 220 V avec deux trous et (souvent) une prise de 
terre. Peu importe ce qu’il y a derrière, du courant alternatif de la centrale du coin, 
un transformateur, quelqu'un qui pédale... nous saurons à coup sûr que nous pouvons 
brancher nos appareils électriques car ces prises s’engagent à nous fournir du courant 
alternatif avec le branchement adéquat. 


Elles respectent le contrat ; elles sont « branchables ». 


Ce dernier terme est un peu barbare et peut faire mal aux oreilles. Mais vous verrez 
que suffixer des interfaces par un « able » est très courant et permet d’être plus précis 
sur la sémantique de l'interface. C’est également un suffixe qui fonctionne en anglais 
et les interfaces du framework .NET finissent pour la plupart par « able ». 


À noter que les interfaces ne fournissent qu’un contrat, elles ne fournissent pas d’im- 
plémentation, c’est-à-dire pas de code C#. Ce n’est pas clair ? Les interfaces indiquent 
que les objets qui choisissent de respecter ce contrat auront forcément telle action 
ou telle caractéristique mais elles n’indiquent pas comment faire, c’est-à-dire qu’elles 


162 


À QUOI SERT LA PROGRAMMATION ORIENTÉE OBJET ? 


n’ont pas de code C## associé. Chaque objet respectant cette interface (on parle d’objet 
implémentant une interface) sera responsable de coder la fonctionnalité associée au 
contrat. 


Pour manipuler ces prises, nous pourrons utiliser cette interface en disant : « Allez 
hop, tous les branchables, venez par ici, on a besoin de votre courant! > Peu importe 
que l’objet implémentant cette interface soit une prise murale ou une prise reliée à 
une dynamo, ou autre, nous pourrons manipuler ces objets par leur interface et donc 
brancher nos prises permettant d'alimenter nos appareils. 


Contrairement à l'héritage, un objet est capable d’implémenter plusieurs interfaces. 
Par exemple, une pompe à chaleur peut être « chauffante » et « refroidissante » !. 


Nous en avons terminé avec la théorie sur les interfaces. Il est fort probable que vous ne 
saisissiez pas encore tout l'intérêt des interfaces ou ce qu’elles sont exactement. Nous 
allons y revenir avec des exemples concrets et vous verrez des utilisations d’interfaces 
dans le cadre du framework .NET qui vous éclaireront davantage. 


À quoi sert la programmation orientée objet ? 


Nous avons décrit plusieurs concepts de la programmation orientée objet mais nous 
n'avons pas encore dit à quoi elle allait nous servir. 


En fait, on peut dire que la POO est une façon de développer une application qui 
consiste à représenter (on dit également « modéliser ») une application informatique 
sous la forme d'objets, ayant des propriétés et pouvant interagir entre eux. 


La modélisation orientée objet est proche de la réalité ce qui fait qu'il sera relativement 
facile de modéliser une application de cette façon. De plus, les personnes non-techniques 
pourront comprendre et éventuellement participer à cette modélisation. 


Cette façon de modéliser les choses permet également de découper une grosse applica- 
tion, généralement floue, en une multitude d’objets interagissant entre eux. Cela permet 
de découper un gros problème en plus petits afin de le résoudre plus facilement. 


Utiliser une approche orientée objet améliore également la maintenabilité. Plus le temps 
passe et plus une application est difficile à maintenir. Il devient difficile de corriger des 
choses sans tout casser ou d’ajouter des fonctionnalités sans provoquer une régression 
par ailleurs. L’orienté objet nous aide ici à limiter la casse en proposant une approche 
où les modifications internes à un objet n’affectent pas tout le reste du code, grâce 
notamment à l’encapsulation. 


Un autre avantage de la POO est la réutilisabilité. Des objets peuvent être réutilisés ou 
même étendus grâce à l'héritage. C’est le cas par exemple de la bibliothèque de classes 
du framework .NET que nous avons déjà utilisée. Cette bibliothèque nous fournit par 
exemple tous les objets permettant de construire des applications graphiques. Inutile 
de réinventer toute la mécanique pour gérer des fenêtres dans une application, le frame- 
work .NET sait déjà faire tout ça. Nous avons juste besoin d'utiliser un objet fenêtre, 


1. Notez qu’en français, nous pourrons également utiliser le suffixe « ante ». En anglais, nous aurons 
plus souvent « able ». 
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dans lequel nous pourrons mettre un objet bouton et un objet zone de texte. Ces 
objets héritent tous des mêmes comportements, comme le fait d’être cliquable ou sé- 
lectionnable, etc. 


De même, des composants tout faits et prêts à l'emploi peuvent être vendus par des 
entreprises tierces (systèmes de log, contrôles utilisateur améliorés, etc.). 


Il faut savoir que la POO, c’est beaucoup plus que ça et nous en verrons des subtilités 
plus loin, mais comprendre ce qu’est un objet est globalement suffisant pour une grande 
partie de l’ouvrage. 


En résumé 


— L'approche orientée objet permet de modéliser son application sous la forme d’inter- 
actions entre objets. 

— Les objets ont des propriétés et peuvent faire des actions. 

— Ils masquent la complexité d’une implémentation grâce à l’encapsulation. 

— Les objets peuvent hériter de fonctionnalités d’autres objets s’il y a une relation 
d’héritage entre eux. 
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Créer son premier objet 


Difficulté : 3 


h , enfin un peu de concret et surtout de code. Dans ce chapitre, nous allons appliquer 
les notions que nous avons vues sur la programmation orientée objet pour continuer 
notre apprentissage du C#. 


Même si vous n'avez pas encore appréhendé exactement où la POO pouvait nous mener, 
ce n'est pas grave. Les notions s’affineront au fur et à mesure de la lecture du livre. Il est 
temps pour nous de commencer à créer des objets, à les faire hériter entre eux, etc. Bref, 
à jouer, grâce à ces concepts, avec le C#. 


Vous verrez qu'avec un peu de pratique tout s'éclaircira ; vous saisirez l'intérêt de la POO 
et comment être efficace avec le C#. 
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Tous les types C-£ sont des objets 


Q Ça y est, il a fini avec son baratin qui donne mal à la tête ? Pourquoi tout ce 


blabla sur les objets ? 


Parce que tout dans le C# est un objet. Comme déjà dit, une fenêtre Windows est un 
objet. Une chaîne de caractères est un objet. La liste que nous avons vue plus haut est 
un objet. 


Nous avons vu que les objets possédaient des caractéristiques ; il s’agit de propriétés. 
Un objet peut également faire des actions ; ce sont des méthodes. 


Suivant ce principe, une chaîne de caractères est un objet et possède des propriétés (par 
exemple sa longueur). De la même façon, il sera possible que les chaînes de caractères 
fassent des actions (par exemple se mettre en majuscules). 


Nous allons voir plus bas qu’il est évidemment possible de créer nos propres objets 
(chat, chien, etc.) et que grâce à eux, nous allons enrichir les types qui sont à notre 
disposition. Un peu comme nous avons déjà fait avec les énumérations. 


Voyons dès à présent comment faire, grâce aux classes. 


Les classes 


Dans le chapitre précédent, nous avons parlé des objets mais nous avons également 
parlé de la définition de l’objet, de sa structure. Eh bien, c’est exactement ce qu'est 
une classe. 


créer des classes. 


Q Une classe est une manière de représenter un objet. Le C# nous permet de 


Nous avons déjà pu voir une classe dans le code que nous avons utilisé précédemment 
et qui a été généré par Visual CÆ Express, la classe Program. Nous n’y avons pas fait 
trop attention, mais voilà à peu près à quoi elle ressemblait : 


1] class Program 

2| { 

3 static void Main(string[] args) 
4 { 

5 } 

6| } 


C’est elle qui contenait la méthode spéciale Main() qui sert de point d’entrée à l’appli- 
cation. Nous pouvons découvrir avec des yeux neufs le mot-clé class qui, comme son 
nom le suggère, permet de définir une classe, c’est-à-dire la structure d’un objet. 
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Rappelez-vous, les objets peuvent avoir des caractéristiques et faire des actions. 


Ici, c'est exactement ce qui se passe. Nous avons défini la structure d’un objet Program 
qui contient une action : la méthode Main(). Vous aurez remarqué au passage que pour 
définir une classe, nous utilisons à nouveau les accolades permettant de créer un bloc 
de code qui délimite la classe. 


Passons sur cette classe particulière et lançons-nous dans la création d’une classe qui 
nous servira à créer des objets. Par exemple, une classe Voiture. À noter qu’il est 
possible de créer une classe à plusieurs endroits dans le code, mais en général, nous 
utiliserons un nouveau fichier, du même nom que la classe, qui lui sera entièrement 
dédié. Une règle d'écriture commune à beaucoup de langages de programmation est 
que chaque fichier doit avoir une seule classe. 


Faisons un clic droit sur notre projet pour ajouter une nouvelle classe, comme indiqué 
à la figure 20.1. 


eee 


[CA Solution 'MaPremiereApplication' (1 projet) 


EU Main(string[] args) 


pipi) 


sl Générer 
Régénérer 
Nettoyer 
à Publier. 
Lä] Nouvel élément... Ctri+Maj+A Ajouter 
E] Élément existant. Maj+Alt+A Ajouter une référence... 
Cis Nouveau dossier Ajouter une référence de service... 
E Formulaire Windows... Définir comme projet de démarrage 
‘Contrôle utilisateur... Déboguer » 
4] Composant. # Couper Ctri+X 
Yg Classe... Maj+at+C [Es Coller CtrisV 


x Supprimer suppr 
Renommer 


3 Propriétés Alt+Entrée 


FIGURE 20.1 — Ajouter une classe 


Visual C# Express nous ouvre sa fenêtre permettant de faire l’ajout d’un élément en 
se positionnant sur l’élément Classe. Nous pourrons donner un nom à cette classe : 
Voiture (voir figure 20.2). 


Vous remarquerez que les classes commencent en général par une majuscule. Visual 
C+ Express nous génère le code suivant : 


using System; 

using System.Collections.Generic; 
using System.Linqg; 

using System.Text; 


namespace MaPremiereApplication 
À 

class Voiture 

{ 

} 


© © I OO Où À CO ND 


EH 
h © 
CA 
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Rechercher Modèles installés 
Éléments Visual C# : £e ; 

2 * Type: Éléments Visual C# 

Classe Éléments Visual C# Lun 
Modèles en ligne Définition de classe vide 

Interface Éléments Visual C# 
Windows Form Éléments Visual C# |E 
Contrôle utilisateur Éléments Visual C# 
Contrôle utilisateur (WPF) Éléments Visual C# 


ge 
à 
(= 
& 
& 
y 
u 


ADO.NET Entity Data Mo. Éléments Visual C# 
ADO.NET EntityObject G... Éléments Visual C# 
Base de données basée s... Éléments Visual C# 
Base de données locale Éléments Visual C# 


Boîte de dialogue À prop. Éléments Visual C# 


Classes LINQ to SQL Éléments Visual C# 


Voiture 


FIGURE 20.2 - Choix d’un modèle de fichier Classe dans la fenêtre de sélection des 
modèles 


Nous retrouvons le mot-clé class suivi du nom de notre classe Voiture et les accolades 
ouvrantes et fermantes permettant de délimiter notre classe. Notons également que 
cette classe fait partie de l’espace de nom MaPremiereApplication qui est l’espace 
de nom par défaut de notre projet. Pour nous simplifier le travail, Visual C# Express 
nous a également inclus quelques espaces de noms souvent utilisés. 


Mais pour l'instant cette classe ne fait pas grand-chose. 


Comme cette classe est vide, elle ne possède ni propriétés ni actions. Nous ne pouvons 
absolument rien faire avec, à part en créer une instance, c’est-à-dire un objet. Cela se 
fait grâce à l’utilisation du mot-clé new. Nous y reviendrons plus en détail plus tard, 
mais cela donne : 


1| static void Main(string[] args) 

2! À 

3 Voiture voitureNicolas = new Voiture(); 
4 Voiture voitureJeremie = new Voiture); 
51 } 


Nous avons créé deux instances de l’objet Voiture et nous les stockons dans les variables 
voitureNicolas et voitureJeremie. 


Si vous vous rappelez bien, nous aurions logiquement dû écrire : 
1| MaPremiereApplication.Voiture voitureNicolas = new 


MaPremiereApplication.Voiture(); 


Ou alors positionner le using qui allait bien, permettant d'inclure l’espace de nom 
MaPremiereApplication. 
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En fait, ici c’est superflu vu que nous créons les objets depuis la méthode Main, qui 
fait partie de la classe Program, faisant partie du même espace de nom que notre classe. 


Les méthodes 


Nous venons de créer notre objet Voiture mais nous ne pouvons pas en faire grand- 
chose. Ce qui est bien dommage. Ça serait bien que notre voiture puisse klaxonner par 
exemple si nous sommes bloqués dans des embouteillages. Bref, que notre voiture soit 
capable de faire des actions. 


Qui dit « action » dit «& méthode ». 


Nous allons pouvoir définir des méthodes faisant partie de notre objet Voiture. Pour 
ce faire, il suffit de créer une méthode, comme nous l’avons déjà vu, directement dans 
le corps de la classe : 


1| class Voiture 

2! À 

3 void Klaxonner() 

4 { 

5 Console.WriteLine("Pouet !"); 
6 } 

713 


Notez quand même l’absence du mot-clé static que nous étions obligés de mettre 
avant. Je vous expliquerai un peu plus loin pourquoi. 


Ce qui fait que si nous voulons faire klaxonner notre voiture, nous aurons juste besoin 
d’invoquer la méthode Klaxonner() depuis l’objet Voiture, ce qui s'écrit : 


1 
2 


Voiture voitureNicolas = new Voiture); 
voitureNicolas.Klaxonner(); 


Cela ressemble beaucoup à ce que nous avons déjà fait. En fait, nous avons déjà uti- 
lisé des méthodes sur des objets. Rappelez-vous, nous avons utilisé la méthode Add() 
permettant d'ajouter une valeur à une liste : 


1 
2 


List<int> maListe = new List<int>(); 
maListe.Add(1); 


Comme nous avons un peu plus de notions désormais, nous pouvons remarquer que 
nous instancions un objet List (plus précisément, une liste d’entier) grâce au mot-clé 
new et que nous invoquons la méthode Add de notre liste pour lui ajouter l’entier 1. 


Nous avons fait pareil pour obtenir un nombre aléatoire, dans une écriture un peu plus 
concise. Nous avions écrit : 


1 | int valeurATrouver = new Random().Next (0, 100); 


Ce qui peut en fait s’écrire : 


1| Random random = new Random); 
2| int valeurATrouver = random.Next (0, 100); 
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Nous créons un objet du type Random grâce à new puis nous appelons la méthode 
Next() qui prend en paramètres les bornes du nombre aléatoire que nous souhaitons 
obtenir (0 étant inclus et 100 exclu). Puis nous stockons le résultat dans un entier. 


Eh oui, nous avons manipulé quelques objets sans le savoir... ! 


Revenons à notre embouteillage et compilons le code nous permettant de faire klaxonner 
notre voiture : 


1| Voiture voitureNicolas = new Voiture); 
2| voitureNicolas.Klaxonner(); 


Impossible de compiler, le compilateur nous indique l’erreur suivante : 


MaPremiereApplication.Voiture.Klaxonner ()’ est inaccessible en 


raison de son niveau de protection 


Diantre! Déjà un premier échec dans notre apprentissage de l’objet ! 


Vous aurez deviné grâce au message d’erreur que la méthode Klaxonner() semble 
inaccessible. Nous expliquerons un peu plus loin de quoi il s’agit. Pour l’instant, nous 
allons juste préfixer notre méthode du mot-clé public, comme ceci : 


1| class Voiture 

21 { 

3 public void Klaxonner() 

4 { 

5 Console.WriteLine("Pouet !"); 
6 } 

713 


Nous allons y revenir juste après, mais le mot-clé public permet d'indiquer que la 
méthode est accessible depuis n’importe où. 


Exécutons notre application et nous obtenons : 


Waouh, la première action d’un de nos objets! 


Bien sûr, ces méthodes peuvent également avoir des paramètres et renvoyer un résultat. 
Par exemple : 


1| class Voiture 

2| { 

3 public bool VitesseAutorisee(int vitesse) 
4 { 

5 if (vitesse > 90) 

6 return false; 

7 else 

8 return true; 

9 + 

10! } 
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Cette méthode accepte une vitesse en paramètre et si elle est supérieure à 90, alors la 
vitesse n’est pas autorisée. Cette méthode pourrait également s’écrire : 


1| class Voiture 

2! { 

3 public bool VitesseAutorisee(int vitesse) 
4 { 

5 return vitesse <= 90; 

6 } 

713 


En effet, nous souhaitons renvoyer faux si la vitesse est supérieure à 90 et vrai si la 
vitesse est inférieure ou égale. 


Donc en fait, nous souhaitons renvoyer la valeur du résultat de la comparaison d’infé- 
riorité ou d'égalité de la vitesse à 90, c’est-à-dire : 


1| class Voiture 

2! { 

3 public bool VitesseAutorisee(int vitesse) 

4 { 

5 bool estVitesseAutorisee = vitesse <= 90; 
6 return estVitesseAutorisee; 

7 } 

8| } 


Ce que nous pouvons écrire finalement en une seule ligne comme précédemment. 


Il est bien sûr possible d’avoir plusieurs méthodes dans une même classe et elles peuvent 
s'appeler entre elles, par exemple : 


1| class Voiture 

2! { 

3 public bool VitesseAutorisee(int vitesse) 
4 { 

5 return vitesse <= 90; 

6 } 

7 

8 public void Klaxonner() 

9 € 
10 if (!VitesseAutorisee(180)) 
11 Console.WriteLine("Pouet !"); 
12 } 
131 } 


Quitte à rouler à une vitesse non autorisée, autant faire du bruit! 
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Notion de visibilité 


Oui, mais le mot-clé public... il nous a bien sauvé la vie, mais... qu'est-ce 
donc ? 


En fait, je l’ai rapidement évoqué et nous nous sommes bien rendu compte que sans ce 
mot-clé, impossible de compiler car la méthode n’était pas accessible. 


Le mot-clé public sert à indiquer que notre méthode pouvait être accessible depuis 
d’autres classes ; en l’occurrence dans notre exemple depuis la classe Program. C’est-à- 
dire que sans ce mot-clé, il est impossible à d’autres objets d'utiliser cette méthode. 


Pour faire en sorte qu’une méthode soit inaccessible, nous pouvons utiliser le mot-clé 
private. Ce mot-clé permet d’avoir une méthode qui n’est accessible que depuis la 
classe dans laquelle elle est définie. Prenons l’exemple suivant : 


1| class Voiture 

2| { 

3 public bool Demarrer() 

4 { 

5 if (ClesSurLeContact ()) 

6 { 

7 DemarrerLeMoteur); 

8 return true; 

9 } 

10 return false; 

11 } 

12 

13 public void SortirDeLaVoiture() 

14 { 

15 if (ClesSurLeContact ()) 

16 PrevenirLUtilisateur(); 

17 } 

18 

19 private bool ClesSurLeContact() 

20 { 

21 // faire quelque chose pour vérifier 
22 return true; 

23 } 

24 

25 private void DemarrerLeMoteur () 

26 { 

27 // faire quelque chose pour démarrer le moteur 
28 } 

29 

30 private void PrevenirLUtilisateur() 
31 { 

32 Console.WriteLine("Bip bip bip"); 
33 } 


NOTION DE VISIBILITÉ 


al} 


Ici seules les méthodes Demarrer() et SortirDeLaVoiture() sont utilisables depuis 
une autre classe, c’est-à-dire que ce sont les seules méthodes que nous pourrons in- 
voquer, car elles sont publiques. Les autres méthodes sont privées à la classe et ne 
pourront être utilisées qu’à l’intérieur de la classe elle-même. Les autres classes n’ont 
pas besoin de savoir comment démarrer le moteur ou comment vérifier que les clés sont 
sur le contact, elles n’ont besoin que de pouvoir démarrer ou sortir de la voiture. Les 
méthodes privées sont exclusivement réservées à l’usage interne de la classe. 


Notez d'ailleurs que la complétion automatique n'est pas proposée pour les 
méthodes inaccessibles. 


Il existe d’autres indicateurs de visibilités que nous allons rapidement décrire : 


Visibilité Description 

public Accès non restreint 

protected Accès depuis la même classe ou depuis une classe 
dérivée 

private Accès uniquement depuis la même classe 

internal Accès restreint à la même assembly 

protected internal Accès restreint à la même assembly ou depuis une 
classe dérivée 


Les visibilités qui vont le plus vous servir sont représentées par les mots clés public et 
private. Nous verrons que le mot-clé protected va servir un peu plus tard quand nous 
parlerons d’héritage. Notez qu’internal pourra être utilisé une fois que nous aurons 
bien maîtrisé toutes les notions. 


Ces mots-clés sont utilisables avec beaucoup d’autres concepts. Nous avons utilisé les 
méthodes pour les illustrer mais ceci est également valable pour les classes ou les 
propriétés que nous allons découvrir juste après. 


Au début, nous avons pu déclarer une classe sans préciser de visibilité... et 
pareil pour la première méthode qui ne compilait pas... c'est normal ? 


Oui, il existe des visibilités par défaut suivant les types déclarés. Vous aurez compris 
par exemple que la visibilité par défaut d’une méthode est privée si l’on ne spécifie pas 
le mot-clé. 


Pour éviter tout risque et toute ambiguïté, il est recommandé de toujours indiquer la 
visibilité ; ce que nous ferons désormais dans ce livre, maintenant que nous savons de 
quoi il s’agit. 
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Les propriétés 


Des objets c’est bien. Des actions sur ces objets, c’est encore mieux. Il nous manque 
encore les caractéristiques des objets. C’est là qu’interviennent les propriétés. 


Sans le savoir, vous avez déjà utilisé des propriétés, par exemple dans le code suivant : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", " 
Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| for (int i = 0; i < jours.Length; i++) 

3| À 

4 Console.WriteLine(jours[il); 

5| } 


Dans la boucle, nous utilisons jours .Length. Nous utilisons en fait la propriété Length 
du tableau « jours », un tableau étant bien sûr un objet. 


Nous avons pu utiliser d’autres propriétés, par exemple dans l’instruction suivante : 


1| List<string> jours = new List<string> { "Lundi", "Mardi", " 
Mercredi", "Jeudi", "Vendredi", "Samedi", "Dimanche" }; 

2| for (int i = 0; i < jours.Count; i++) 

3| + 

4 Console.WriteLine(jours[il); 

5[} 


Ici, Count est une propriété de la liste « jours ». 


De la même façon, nous avons la possibilité de créer des propriétés sur nos classes pour 
permettre d'ajouter des caractéristiques à nos objets. 


Par exemple, nous pouvons rajouter les propriétés suivantes à notre voiture : une 
couleur, une marque, une vitesse. Il y a plusieurs façons de rajouter des caractéristiques 
à un objet. La première est d'utiliser des variables membres. 


Les variables membres 


Ici en fait, un objet peut avoir une caractéristique sous la forme d’une variable publique 
qui fait partie de la classe. Pour ce faire, il suffit de définir simplement la variable à 
l’intérieur des blocs de code délimitant la classe et de lui donner la visibilité public. 


Rajoutons par exemple une chaîne de caractères permettant de stocker la couleur de 
la voiture : 


1| public class Voiture 

2| € 

3 public string Couleur; 
al} 


Notez que j'ai rajouté les visibilités pour la classe et pour la variable. 


Grâce à ce code, la chaîne de caractères Couleur est désormais une caractéristique de 
la classe Voiture. Nous pourrons l’utiliser en faisant suivre notre objet de l’opérateur 
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«.» suivi du nom de la variable. Ce qui donne : 


1| Voiture voitureNicolas = new Voiture); 
2| voitureNicolas.Couleur = "rouge; 
3| Voiture voitureJeremie = new Voiture(); 
4| voitureJeremie.Couleur = "verte"; 


Cela ressemble beaucoup à ce que nous avons déjà fait. Nous utilisons le «.» pour 
accéder aux propriétés d’un objet. Comme d’habitude, les variables vont pouvoir stocker 
des valeurs. Ainsi, nous pourrons avoir une voiture rouge pour Nicolas et une voiture 
verte pour Jérémie. 


Notez ici qu'il s'agit bien de variables membres et non de vraies propriétés. 
En général, les variables d’une classe ne doivent jamais être publiques. Nous 
utiliserons rarement cette construction. 


Les propriétés : 


Les propriétés sont en fait des variables évoluées. Elles sont à mi-chemin entre une 
variable et une méthode. 


Prenons cet exemple : 


1| public class Voiture 
2! { 
3 private int vitessePrivee; 
4 public int Vitesse 
5 { 
6 get 
4 { 
8 return vitessePrivee; 
9 } 
10 set 
11 { 
12 vitessePrivee = value; 
13 } 
14 } 
151 } 


Nous pouvons voir que nous définissons dans un premier temps une variable privée, 
vitessePrivee de type entier. Comme prévu, cette variable est masquée pour les 
utilisateurs d'objets Voiture. Ainsi, le code suivant : 


1| Voiture voitureNicolas = new Voiture); 
2| voitureNicolas.vitessePrivee = 50; 


provoquera l'erreur de compilation désormais bien connue : 


MaPremiereApplication.Voiture.vitessePrivee”? est inaccessible en 


raison de son niveau de protection 
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Par contre, nous en avons profité pour définir la propriété Vitesse, de type int. 
Pour ceci, nous avons défini une partie de la propriété avec le mot-clé get suivi d’un 
return vitessePrivee. Et juste en dessous, nous avons utilisé le mot-clé set, suivi 
de vitessePrivee = value. 


Ce que nous avons fait, c’est définir la possibilité de lire la propriété grâce au mot- 
clé get. Ici, la lecture de la propriété nous renvoie la valeur de la variable privée. De 
la même façon, nous avons défini la possibilité d’affecter une valeur à la propriété en 
utilisant le mot-clé set et en affectant la valeur à la variable privée. 


Les blocs de code délimités par get et set se comportent un peu comme des méthodes. 
Ils ont un corps qui est délimité par des accolades et dans le cas du get, ils doivent 
renvoyer une valeur du même type que la propriété. 


À noter que dans le cas du set, « value » est un mot-clé qui permet de dire : «la 
valeur que nous avons affectée à la propriété ». 


Prenons l’exemple suivant : 


1| Voiture voitureNicolas = new Voiture); 
2| voitureNicolas.Vitesse = 50; 
3| Console.WriteLine(voitureNicolas.Vitesse); 


La première instruction instancie bien sûr une voiture. 


La deuxième instruction consiste à appeler le bloc de code set de Vitesse qui met la 
valeur 50 dans la pseudo-variable value qui est stockée ensuite dans la variable privée. 


Lorsque nous appelons la troisième instruction, nous lisons la valeur de la propriété et 
pour ce faire, nous passons par le get qui nous renvoie la valeur de la variable privée. 


Ok, c'est super, mais dans ce cas, pourquoi passer par une propriété et pas 
par une variable ? Même s'il paraît que les variables ne doivent jamais être 
publiques... 


Eh bien parce que dans ce cas-là, la propriété peut faire un peu plus que simplement 
renvoyer une valeur. Et aussi parce que nous masquons la structure interne de notre 
classe à ceux qui veulent l’utiliser. Nous pourrions très bien envisager d’aller lire la 
vitesse dans les structures internes du moteur, ou en faisant un calcul poussé par 
rapport au coefficient du vent et de l’âge du capitaine (ou plus vraisemblablement en 
allant lire la valeur en base de données). Et ici, nous pouvons tirer parti de la puissance 
des propriétés pour masquer tout ça à l’appelant qui, lui, n’a besoin que d’une vitesse. 


Par exemple : 


1| private int vitessePrivee; 

2| public int Vitesse 

3| + 

4 get 

5 { 

6 int v = vitesseDesRoues * rayon * coefficient; // ce 
calcul est complètement farfelu ! 

7 MettreAJourLeCompteur () ; 
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8 AdapterLaVitesseDesEssuieGlaces(); 

9 return v; 

10 } 

ER set 

12 { 

13 // faire la mise à jour des variables internes 
14 MettreAJourLeCompteur (); 

15 AdapterLaVitesseDesEssuieGlaces(); 

16 } 

17 | } 


En faisant tout ça dans le bloc de code get, nous masquons les rouages de notre classe 
à l'utilisateur. Lui, il n’a besoin que d’obtenir la vitesse, sans s’encombrer du compteur 
ou des essuie-glaces. 


Bien sûr, la même logique peut s'adapter au bloc de code set, qui permet d’affecter 
une valeur à la propriété. 


Il est également possible de rendre une propriété en lecture seule, c'est-à-dire non 
modifiable par les autres objets. Il pourra par exemple sembler bizarre de positionner 
une valeur à la vitesse alors qu’en fait, pour mettre à jour la vitesse, il faut appeler la 
méthode AppuyerPedale (double forceAppui). 


Pour empêcher les autres objets de pouvoir directement mettre à jour la propriété 
Vitesse, il suffit de ne pas déclarer le bloc de code set et de ne garder qu’un get. Par 
exemple : 


1| public class Voiture 

2| { 

3 private int vitessePrivee; 

4 public int Vitesse 

5 { 

6 get 

7 L 

8 // faire des calculs 
9 return vitessePrivee; 
10 } 

11 } 

1217} 


Ce qui fait que si nous tentons d’affecter une valeur à la propriété Vitesse depuis une 
autre classe, par exemple : 


1| Voiture voitureNicolas = new Voiture); 
50; 


2| voitureNicolas.Vitesse 
Nous aurons l’erreur de compilation suivante : 


La propriété ou l’indexeur ?’MaPremiereApplication.Voiture. 


Vitesse”? ne peut pas être assigné -- il est en lecture 
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Le compilateur nous indique donc très justement qu’il est impossible de faire cette 
affectation car la propriété est en lecture seule. Il devient donc impossible pour un 
utilisateur de cette classe de modifier la vitesse de cette façon. 


De même, il est possible de définir une propriété pour qu’elle soit accessible en écriture 
seule. Il suffit de définir uniquement le bloc de code set : 


private double acceleration; 
public double Acceleration 
{ 


1 

2 

3 

4 set 

5 { 

6 acceleration = value; 
à } 

8| } 


Ainsi, si nous tentons d'utiliser la propriété en lecture, avec par exemple : 


1 | Console.WriteLine(voitureNicolas.Acceleration); 
Nous aurons l'erreur de compilation attendue : 


La propriété ou l’indexeur ?’MaPremiereApplication.Voiture. 


Acceleration?” ne peut pas être utilisé dans ce contexte, car 
il lui manque l’accesseur get 


Ce bridage d’accès à des propriétés prend tout son sens quand nous souhaitons exposer 
nos objets à d’autres consommateurs qui n’ont aucun intérêt à connaître la structure 
interne de notre classe. C’est un des principes de l’encapsulation. 


Les propriétés auto-implémentées 


Les propriétés auto-implémentées sont une fonctionnalité que nous allons beaucoup 
utiliser. Il s’agit de la définition d’une propriété de manière très simplifiée qui va nous 
servir dans la grande majorité des cas où nous aurons besoin d’écrire des propriétés. 
Dans cet ouvrage, nous l’utiliserons très souvent. 


Ainsi, le code suivant que nous avons déjà vu : 


1| private int vitesse; 

2| public int Vitesse 

31 € 

< get 

5 { 

6 return vitesse; 
dé de 

8 set 

9 { 

10 vitesse = value; 
11 } 

12| } 
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est un cas très fréquent d'utilisation de propriétés. Nous exposons ici une variable privée 
à travers les propriétés get et set. L'écriture du dessus est équivalente à la suivante : 


1 | public int Vitesse { get; set; } 

Dans ce cas, le compilateur fait le boulot lui-même, il génère (dans le code compilé) 
une variable membre qui va servir à stocker la valeur de la propriété. 

C’est exactement pareil, sauf que cela nous simplifie grandement l’écriture. 


En plus, nous pouvons encore accélérer son écriture en utilisant ce qu’on appelle des 
« snippets » qui sont des extraits de code. Pour les utiliser, il suffit de commencer 
à écrire un mot et Visual C# nous complète le reste. Un peu comme la complétion 
automatique sauf que cela fonctionne pour des bouts de code très répétitifs et très 
classiques à écrire. 


Commencez par exemple à taper « prop » ; Visual C# nous propose plusieurs extraits 
de code, comme vous pouvez le voir sur la figure 20.3. 


© Là IMIGF TENTE EAP PIC GLIUTI, VUILUTE % 


using System.Collections.Generic; 
using System.Ling; 
using System.Text; 


E | MaPremiereApplication 


public class Voiture 


{ 
BroP 
—— prop 
am propa Extrait de code pour une propriété implémentée automatiquement 
à propdp Version de langage : C# 3.0 ou supérieure 
A propfull 
à propg 


FIGURE 20.3 - Utilisation du snippet de création de propriété auto-implémentée 


Appuyez sur | Tab | ou | Entrée | pour sélectionner cet extrait de code et appuyez ensuite 


sur pour que Visual C# génère l'extrait de code correspondant à la propriété 
auto-implémentée (voir figure 20.4). 


using System.Collections.Generic; 
using System.Ling; 
using System.Text; 


Enamespace MaPremiereApplication 
E public class Voiture 


public EME MyP 


ty { get; set; } 


FIGURE 20.4 - Changer le type de la propriété 


Ici, int est surligné et vous pouvez, si vous le souhaitez, changer le type de la propriété, 
par exemple string. Appuyez à nouveau sur et Visual C# surligne le nom de 
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la propriété, que vous pouvez à nouveau changer. Appuyez enfin sur pour 
terminer la saisie et vous aurez une belle propriété auto-implémentée (voir figure 20.5). 


using System.Text; 


=inamespace MaPremiereApplication 


public string Couleur { get; set; } 


{ 
E public class Voiture 
{ 
} 
} 


FIGURE 20.5 — La propriété auto-implémentée 


Vous avez pu remarquer qu’en commençant à taper « prop», Visual C# Express vous a 
proposé d’autres extraits de code, par exemple « propfull » qui va générer la propriété 
complète telle qu’on l’a vue un peu plus haut. 


D’autres extraits de code existent, comme « propg » qui permet de définir une propriété 
en lecture seule. 


En effet, comme au chapitre précédent, il est possible de définir des propriétés auto- 
implémentées en lecture seule ou en écriture seule avec une écriture simplifiée. Dans le 
cas des propriétés auto-implémentées, il y a cependant une subtilité. 


Pour avoir de la lecture seule, nous devons indiquer que l'affectation est privée, comme 
on peut le voir en utilisant l’extrait de code « propg ». Visual C#Æ Express nous génère 
le bout de code suivant : 


1 | public int Vitesse { get; private set; } 


En positionnant une propriété d'écriture en privé, Visual C# Express autorise la classe 
Voiture à modifier la valeur de Vitesse, que ce soit par une méthode ou par une 
propriété. 

À noter que si nous n'avions pas mis private set et que nous avions simplement 
supprimé le set, alors la compilation aurait été impossible. En effet, il s'avère difficile 
d'exploiter une propriété auto-implémentée en lecture alors que nous n’avons pas la 
possibilité de lui donner une valeur. Ou inversement. 


Alors, pourquoi nous avoir parlé de la possibilité de complètement supprimer 
la lecture ou l'écriture ? Ce n'est pas plus intéressant de toujours mettre la 
propriété en private? 


Si c’est une propriété auto-implémentée, évidemment que si. Par contre, si nous n’uti- 
lisons pas une propriété auto-implémentée et que nous utilisons une variable membre 
pour sauvegarder la valeur de la propriété, nous pourrons éventuellement modifier ou 
lire la valeur de la variable à partir d’une méthode ou d’une autre propriété. 


Bref, maintenant que vous connaissez les deux syntaxes, vous pourrez utiliser la plus 
adaptée à votre besoin. 
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Voilà pour les propriétés. 


À noter que quand nous avons beaucoup de propriétés à initialiser sur un objet, nous 
pouvons utiliser une syntaxe plus concise. Par exemple, les instructions suivantes : 


1| Voiture voitureNicolas = new Voiture); 
2 

3| voitureNicolas.Couleur = "Bleue; 

4| voitureNicolas.Marque = "Peugeot"; 

5| voitureNicolas.Vitesse = 50; 


sont équivalentes à l’instruction : 


1| Voiture voitureNicolas = new Voiture { Couleur = "Bleue", 
Marque = "Peugeot", Vitesse = 50 }; 


Les accolades servent ici à initialiser les propriétés au même moment que l’instanciation 
de l’objet. 


Note : cela paraît évident, mais il est bien sûr possible d'accéder aux propriétés 
d'une classe depuis une méthode de la même classe. 


En résumé 


— On utilise des classes pour représenter quasiment la plupart des objets. 
— On utilise le mot-clé class pour définir une classe et le mot-clé new pour l’instancier. 


— Une classe peut posséder des caractéristiques (les propriétés) et peut faire des actions 
(les méthodes). 


181 


CHAPITRE 20. CRÉER SON PREMIER OBJET 


182 


a 


Manipuler des objets 


Difficulté : 3 


aintenant que nous avons bien vu comment définir des objets, il est temps de 

savoir les utiliser. Nous allons voir dans ce chapitre quelles sont les subtilités de 

l'instanciation des objets. Vous verrez notamment ce qu'est un constructeur et qu'on 
peut avoir des valeurs nulles pour des objets. N'hésitez pas à jouer avec tous ces concepts, 
il est important d'être à l'aise avec les bases pour pouvoir être efficace avec les concepts 
avancés | 
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Le constructeur 


Le constructeur est une méthode particulière de l’objet qui permet de faire des choses 
au moment de la création d’un objet, c’est-à-dire au moment où nous utilisons le mot- 
clé new. Il est en général utilisé pour initialiser des valeurs par défaut d’un objet. Par 
exemple, si nous voulons que lors de la création d’une voiture, elle ait automatiquement 
une vitesse, nous pouvons faire : 


1| public class Voiture 

2| € 

3 public int Vitesse { get; set; } 
4 

5 public Voiture() 

6 { 

7 Vitesse = 5; 

8 } 

9} 


Le constructeur est en fait une méthode spéciale qui a le même nom que la classe et 
qui ne possède pas de type de retour. Elle est appelée lors de la création de l’objet, 
avec new. 


Pour illustrer le comportement du constructeur, ajoutons une méthode Rouler à notre 
classe, de cette façon : 


1| public class Voiture 

2| { 

3 public int Vitesse { get; set; } 
4 

5 public Voiture() 

6 { 

7 Vitesse = 5; 

8 } 

9 

10 public void Rouler() 

11 { 

12 Console.WriteLine ("Je roule à " + Vitesse + " km/h"); 
13 } 

|} 


Que nous pourrons appeler ainsi : 


Voiture voitureNicolas = new Voiture); 
voitureNicolas.Rouler(); 
voitureNicolas.Vitesse 


50; 


& © NN H 


voitureNicolas.Rouler); 


Au moment de l’instanciation de l’objet avec new, la vitesse va être égale à 5. Nous 
faisons rouler la voiture. Puis nous changeons la vitesse pour la passer à 50 et nous 
faisons à nouveau rouler la voiture : 
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Je roule 5 km/h 


Je roule 50 km/h 


Le constructeur est la première « méthode » à être appelée lors de la création d’un objet. 
C’est l'endroit approprié pour faire des initialisations, ou pour charger des valeurs, etc. 


Le constructeur que nous avons vu est ce qu’on appelle le constructeur par défaut, 
car il ne possède pas de paramètres. Il est possible de passer des paramètres à un 
constructeur, pour initialiser des variables de notre classe avec des valeurs. Pour ce 
faire, nous devons déclarer un constructeur avec un paramètre. Par exemple : 


1| public class Voiture 

2! { 

3 public int Vitesse { get; set; } 
4 

5 public Voiture) 

6 € 

7 Vitesse = 5; 

8 } 

9 

10 public Voiture(int vitesse) 
11 { 
12 Vitesse = vitesse; 
13 } 
14 
15 public void Rouler() 
16 { 
17 Console.WriteLine("Je roule à " + Vitesse + " km/h"); 
18 } 
191 } 


Ainsi, nous pourrons créer un objet voiture en lui précisant une vitesse par défaut de 
cette façon : 


1 | Voiture voitureNicolas = new Voiture (20); 


Après ceci, la variable voitureNicolas aura une vitesse de 20. 


Bien sûr, nous pourrions faire la même chose sans utiliser de constructeur, en affectant 
une valeur à la propriété après avoir instancié l’objet. Ce qui fonctionnerait tout à fait. 
Sauf qu'il ne se passe pas exactement la même chose. Le constructeur est vraiment ap- 
pelé en premier, dès qu’on utilise new pour créer un objet. Les propriétés sont affectées 
fatalement après, donc tout dépend de ce que l’on veut faire. 

Donc, oui, on pourrait affecter une valeur à des propriétés pour faire ce genre d’ini- 
tialisation juste après avoir instancié notre objet mais cela nous oblige à écrire une 
instruction supplémentaire qui pourrait ne pas paraître évidente ou obligatoire. 


n S ûre av nstructeur ue nous somm igé sser 1, 
Une chose est sûre avec le constructeur, c’est que nous sommes obligés d’y passer et ceci, 
peu importe la façon dont on utilise la classe. L’initialisation devient donc obligatoire, 
on évite le risque qu’une propriété soit nulle. 
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À noter qu’il est possible de cumuler les constructeurs tant qu'ils ont chacun des para- 
mètres différents. Dans notre exemple, nous pourrons donc créer des voitures de deux 
façons différentes : 


1| Voiture voitureNicolas new Voiture(); // vitesse vaut 5 


new Voiture(20); // vitesse vaut 20 


2| Voiture voitureJeremie 


Il est aussi possible de ne pas définir de constructeur par défaut et d’avoir unique- 
ment un constructeur possédant des paramètres. Dans ce cas, il devient impossible 
d’instancier un objet sans lui passer de paramètres. 


Instancier un objet 


N 2 


Nous allons revenir à présent sur l’instanciation d’un objet. Comme nous venons de 
le voir, nous utilisons le mot-clé new pour créer une instance d’un objet. C’est lui 
qui permet la création d’un objet. Il appelle le constructeur correspondant. Si aucun 
constructeur n'existe, il ne se passera rien de plus qu’une création de base. Par exemple : 


1 | Voiture voitureNicolas = new Voiture); 


Pour rentrer un peu dans la technique, au moment de l’instanciation d’un objet, l’opé- 
rateur new crée l’objet et le met à une place disponible en mémoire. Cette adresse 
mémoire est conservée dans la variable voitureNicolas. On dit que voitureNicolas 
contient une référence vers l’objet. Nous verrons un peu plus tard ce que cela implique. 


Ce principe ressemble un peu au « pointeur >» que nous pourrions trouver 
dans des langages comme le C ou le C++. Typiquement, si vous savez ce 
qu'est un pointeur, vous pouvez vous représenter une référence comme un 
pointeur évolué. 


Comme pour les types que nous avons vus plus haut, nous sommes obligés d’initialiser 
un objet avant de l'utiliser. Sinon, Visual C# Express nous générera une erreur de 
compilation. Par exemple, les instructions suivantes : 


1 
2 


Voiture voitureNicolas; 
voitureNicolas.Vitesse = 5; 


provoqueront l’erreur de compilation suivante : 


Utilisation d’une variable locale non assignée ?’voitureNicolas”? 


En effet, Visual C# Express est assez intelligent pour se rendre compte que l’on va 
essayer d'accéder à un objet qui n’a pas été initialisé. 


Il faut toujours initialiser une variable avant de pouvoir l'utiliser. Comme pour les types 
précédents, il est possible de dissocier la déclaration d’un objet de son instanciation, 
en écrivant les instructions sur plusieurs lignes, par exemple : 
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1| Voiture voitureNicolas; 
2| // des choses 
3| voitureNicolas = new Voiture); 


ceci est possible tant que nous n’utilisons pas la variable voitureNicolas avant de 
l’avoir instanciée. 


Les objets peuvent également avoir une valeur nulle. Ceci est différent de l’absence 
d’initialisation car la variable est bien initialisée et sa valeur vaut « nul ». Ceci est 
possible grâce à l'emploi du mot-clé null. 


1 | Voiture voitureNicolas = null; 


N 


Attention, il est par contre impossible d'accéder à un objet qui vaut null. Eh oui, 
comment voulez-vous vous asseoir sur une chaise qui n’existe pas ? Eh bien vous vous 
retrouverez avec les fesses par terre, personne ne vous a indiqué que la chaise n'existait 
pas. 


C’est pareil pour notre application, si nous tentons d'utiliser une voiture qui n’existe 
pas, nous aurons droit à un beau plantage. 


Par exemple, avec le code suivant : 


1] Voiture voitureNicolas = null; 
2| voitureNicolas.Vitesse bi 


vous n’aurez pas d’erreur à la compilation, par contre vous aurez un message d’excep- 
tion assez explicite : 


Exception non gérée : System.NullReferenceException: La réfé 
rence d’objet n’est pas définie à une instance d’un objet. 
à MaPremiereApplication.Program.Main(String[] args) dans C:\ 


Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
13 


Comme nous l’avons déjà vu, le programme nous affiche une exception; nous avons 
dit que c'était simplement une erreur qui faisait planter notre programme. Ce n’est 
pas bien! Surtout que cela se passe au moment de l’exécution. Nous perdons toute 
crédibilité ! 

Ici, le programme nous dit que la référence d’un objet n’est pas définie à une instance 
d’un objet. Concrètement, cela veut dire que nous essayons de travailler sur un objet 
null. 


Pour éviter ce genre d’erreur à l'exécution, il faut impérativement instancier ses objets, 
en utilisant l’opérateur new, comme nous l’avons déjà vu. Il n’est cependant pas toujours 
pertinent d’instancier un objet dont on pourrait ne pas avoir besoin. Le C# nous 
offre donc la possibilité de tester la nullité d’un objet. Il suffit d'utiliser l’opérateur de 


comparaison & —— » en comparant un objet au mot-clé null, par exemple : 
1] string prenom = "Nicolas"; 
2| Voiture voiture = null; 
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3| if (prenom == "Nicolas") 

4 voiture = new Voiture { Vitesse = 50 }; 

5| if (voiture == null) 

6| { 

7 Console.WriteLine ("Vous n'avez pas de voiture"); 
8| } 

9 | else 

10 | { 

11 voiture.Rouler(); 

12| } 


Ainsi, seul Nicolas possédera une voiture et le test de nullité sur l’objet permet d'éviter 
une erreur d'exécution si le prénom est différent. 


Maintenant que vous connaissez le mot-clé null et que vous savez qu’un objet peut 
prendre une valeur nulle, nous allons revenir sur un point que j'ai rapidement abordé 
auparavant. Je ne sais pas si vous vous en rappelez, mais lors de l’étude des opérateurs 
logiques j'ai parlé du fait que l’opérateur OÙ ( || ) évaluait la première condition et 
si elle était vraie alors il n’évaluait pas la suivante, considérant que de toute façon, le 
résultat allait être vrai. 


Ce détail prend toute son importance dans le cas suivant : 


if (voiture == null || voiture.Couleur == "Bleue") 
{ 


// faire quelque chose 
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} 


Dans ce cas, si la voiture est effectivement nulle, alors le fait d'évaluer la propriété 
Couleur de la voiture devrait renvoyer une erreur. Heureusement, le C# avait prévu 
le coup. Si la première condition est vraie alors la seconde ne sera pas évaluée, ce qui 
évitera l'erreur. Ainsi, nous sommes sûrs de n’avoir aucune voiture bleue. 


Il est par contre évident qu’une telle condition utilisant l'opérateur ET (&&) est une 
hérésie car pour que la condition soit vraie, le C# a besoin d’évaluer les deux opérandes. 
Et donc si la voiture est nulle, l’utilisation d’une propriété sur une valeur nulle renverra 
une erreur. 


Notons également que lorsque nous utilisons l’opérateur ET (&&), si la première opé- 
rande est fausse, alors de la même façon, il n'évalue pas la seconde, car pour que la 
condition soit vraie il faut que les deux le soient. Ce qui fait qu’il est également possible 
d'écrire ce code : 


1| if (voiture != null && voiture.Couleur == "Rouge") 
2| { 

3 // faire autre chose 

al} 


qui ne provoquera pas d’erreur à l’exécution, même si voiture vaut null car dans ce 
cas, le fait que le premier test soit faux évitera le test de l’autre partie de l’expression. 


Vous verrez que vous aurez l’occasion d'utiliser le mot-clé null régulièrement. 


188 


LE MOT-CLÉ THIS 


Le mot-clé this 


Lorsque nous écrivons le code d’une classe, le mot-clé this représente l’objet dans 
lequel nous nous trouvons. Il permet de clarifier éventuellement le code, mais il est gé- 
néralement facultatif. Ainsi, pour accéder à une variable de la classe ou éventuellement 
une méthode, nous pouvons les préfixer par « this. ». Par exemple, nous pourrions 
écrire notre classe de cette façon : 


1| public class Voiture 

2! { 

3 public int Vitesse { get; set; } 

4 public string Couleur { get; set; } 

5 

6 public Voiture) 

7 { 

8 this.Vitesse = 5; 

9 } 

10 

11 public void Rouler() 

12 { 

13 Console.WriteLine("Je roule à " + this.Vitesse + " km/h 
ns 

14 } 

15 

16 public void Accelerer(int acceleration) 

17 { 

18 this.Vitesse += acceleration; 

19 this.Rouler(); 

20 } 

21| } 


Ici, dans le constructeur, nous utilisons le mot-clé this pour accéder à la propriété 
Vitesse. C’est la même chose dans la méthode Rouler. De la même façon, on peut uti- 
liser this.Rouler() pour appeler la méthode Rouler depuis la méthode Accelerer(). 


C’est une façon pour la classe de dire : « Regardez, avec this, c’est "ma variable à 


moi" ». 


Notez bien sûr que sans le mot-clé this, notre classe compilera quand même et sera 
tout à fait fonctionnelle. Ce mot-clé est facultatif mais il peut aider à bien faire la 
différence entre ce qui appartient à la classe et ce qui fait partie des paramètres des 
méthodes ou d’autres objets utilisés. 


Suivant les personnes, le mot-clé this est soit systématiquement utilisé, soit jamais. Je 
fais plutôt partie des personnes qui ne l’utilisent jamais. Il arrive par contre certaines 
situations où il est absolument indispensable, comme celle-ci, mais en général, j'essaie 
d'éviter ce genre de construction : 


1| public void ChangerVitesse(int Vitesse) 
2| { 


à this.Vitesse = Vitesse; 
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al} 


Vous remarquerez que le paramètre de la méthode ChangerVitesse() et la propriété 
ou variable membre de la classe ont exactement le même nom. Ceci est possible ici 
mais source d'erreurs, les variables ayant des portées différentes. Il s’avère que dans ce 
cas, le mot-clé this est indispensable. On pourra donc éviter l'ambiguïté en préfixant 
la propriété membre avec le mot-clé this. 


Je recommande plutôt de changer le nom du paramètre, quitte à utiliser une minuscule, 
ce qui augmentera la lisibilité et évitera des erreurs potentielles. 


En général, des conventions de nommage pourront nous éviter de nous retrouver dans 
ce genre de situation. 


Ça y est, nous savons instancier et utiliser des objets ! Savoir créer et utiliser ses propres 
objets est très important dans un programme orienté objet. Vous allez également avoir 
besoin très régulièrement d’utiliser des objets tout faits, comme ceux venant de la bi- 
bliothèque de classe du framework .NET. Nous comprenons d’ailleurs mieux pourquoi 
elle s’appelle « bibliothèque de classes ». Il s’agit bien d’un ensemble de classes utili- 
sables dans notre application et nous pourrons instancier les objets relatifs à ces classes 
pour nos besoins. Comme ce que nous avions déjà fait auparavant sans trop le savoir, 
avec l’objet Random par exemple... 


N'hésitez pas à relire ce chapitre ainsi que le précédent si vous n’avez pas parfaitement 
compris toutes les subtilités de la création d'objet. C’est un point important du livre. 


En résumé 


— Les classes possèdent une méthode particulière, appelée à l’instanciation de l’objet : 
le constructeur. 

— Une instance d’une classe peut être initialisée avec une valeur nulle grâce au mot-clé 
null. 

— Le mot-clé this représente l’objet en cours de la classe. 
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RE 


La POO et le CÆ 


Difficulté : (3 


ans ce chapitre, vous allez vous immerger un peu plus dans les subtilités de la POO 

en utilisant le C#. Il est temps un peu de tourmenter nos objets et de voir ce qu'ils 

ont dans le ventre. Ainsi, nous allons voir comment les objets héritent les uns des 
autres où comment fonctionnent les différents polymorphismes. 


Nous allons également voir comment tous ces concepts se retrouvent dans le quotidien d'un 


développeur C#. 
À 
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Des types, des objets, type valeur et type référence 


Ok, je sais maintenant créer des objets, mais je me rappelle qu'au début du 
livre, nous avons manipulé des int et des string et que tu as appelé ça des 
« types » ; et après, tu nous dis que tout est objet... Tu serais pas en train 
de raconter n'importe quoi par hasard ? | 


Eh bien non, Ô perspicace lecteur! Précisons un peu, maintenant que vous avez de 
meilleures connaissances. J’ai bien dit que tout était objet, je le maintiens, même sous 
la torture! C'est-à-dire que même les types simples comme les entiers int ou les chaînes 
de caractères sont des objets. 


J’en veux pour preuve ce simple exemple : 


Console.WriteLine(chaineEnMajuscule); 
Console.WriteLine(chaineEnMajuscule.Length) ; 


1] int a = 10; 

2| string chaine = a.ToString(); 

3| chaine = "abc" + chaine; 

4| string chaineEnMajuscule = chaine.ToUpper(); 
5 

6 


La variable a est un entier. Nous appelons la méthode ToString() sur cet entier. Même 
si nous n’avons pas encore vu à quoi elle servait, nous pouvons supposer qu’elle effectue 
une action qui consiste à transformer l’entier en chaîne de caractères. Nous concaténons 
ensuite la chaîne abc à cette chaîne et nous effectuons une action qui, à travers la mé- 
thode ToUpper (), met la chaîne en majuscule. Enfin, la méthode Console .WriteLine 
nous affiche « ABC10 » puis nous affiche la propriété Length de la chaîne de caractères 
qui correspond bien sûr à sa taille. 


Pour créer une chaîne de caractères, nous utilisons le mot-clé string. Sachez que ce 
mot-clé est équivalent à la classe String (notez la différence de casse). En créant une 
chaîne de caractères, nous avons instancié un objet défini par la classe String. 


Mais alors, pourquoi utiliser string et non pas String ? 


En fait, le mot-clé string est ce qu'on appelle un alias de la classe String qui se situe 
dans l’espace de nom System. De même, le mot-clé int est un alias de la structure 
Int32 qui se situe également dans l’espace de nom System (nous verrons un peu plus 
loin ce qu'est vraiment une structure). 


Ce qui fait que les instructions suivantes : 
1] int a = 10; 

2| string chaine = "abc"; 

sont équivalentes à celles-ci : 


1| System.Int32 a = 10; 
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2 | System.String chaine = "abc; 


Q En pratique, comme on l'a déjà fait, on utilise plutôt les alias que les classes 


qu'ils représentent. 


Cependant, les entiers, les booléens et autres types « simples » sont ce qu’on appelle des 
types intégrés. Et même si ce sont des objets à part entière (méthodes, propriétés... ), 
ils ont des particularités, notamment dans la façon dont ils sont gérés par le framework 
.NET. On les appelle des types valeur, car les variables de ce type possèdent la vraie 
valeur de ce qu’on leur affecte a contrario des classes qui sont des types référence 
dont les variables possèdent simplement un lien vers un objet en mémoire. 


Par exemple : 


1 | int entier = 5; 


Ici, la variable contient vraiment l’entier 5. Alors que pour l’instanciation suivante : 


1 | Voiture voitureNicolas = new Voiture); 


La variable voitureNicolas contient une référence vers l’objet en mémoire. 


On peut imaginer que le type référence est un peu comme si on disait que ma maison 
se situe au « 9 rue des bois ». L'adresse à été écrite sur un bout de papier et référence 
ma maison qui ne se situe bien sûr pas au même endroit que le bout de papier. Si je 
veux vraiment voir l’objet maison, il va falloir que j'aille voir où c’est indiqué sur le 
bout de papier. C’est ce que fait le type référence, il va voir en mémoire ce qu'il y a 
vraiment dans l’objet. 


Alors que le type valeur pourrait ressembler à un billet de banque par exemple. Je 
peux me balader avec, il est marqué 500 € dessus (oui, je suis riche!) et je peux payer 
directement avec sans que le fait de donner le billet implique d’aller chercher le contenu 
à la banque. 


— Le type valeur contient la vraie valeur qui en général est assez petite et facile à 
stocker. 
— Le type référence ne contient qu'un lien vers un plus gros objet stocké ailleurs. 


Cette manière différente de gérer les types et les objets implique plusieurs choses. Dans 
la mesure où les types valeur possèdent vraiment la valeur de ce qu’on y stocke, une 
copie de la valeur est effectuée à chaque fois que l’on fait une affectation. C’est possible 
car ces types sont relativement petits et optimisés. Cela s'avère impossible pour un 
objet qui est trop gros et trop complexe. C’est un peu compliqué de copier toute ma 
maison alors que c’est un peu plus simple de recopier ce qu’il y a sur le bout de papier. 


Ainsi, l'exemple suivant : 


1] int a = 5; 

2| int b = a; 

3| b = 6; 

4| Console.WriteLineKa); 
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5 | Console.WriteLine(b); 


affichera les valeurs 5 puis 6; ce qui est le résultat que l’on attend. 


— En effet, la variable a à été initialisée à 5. 
— On a ensuite affecté a à b. La valeur 5 s’est copiée (dupliquée) dans la variable b. 
— Puis nous avons affecté 6 à b. 


Ce qui paraît tout à fait logique ! Par contre, l’exemple suivant : 


Console.WriteLine(voitureNicolas.Couleur); 
Console.WriteLine (voitureJeremie.Couleur); 


1| Voiture voitureNicolas = new Voiture); 
2| voitureNicolas.Couleur = "Bleue"; 

3| Voiture voitureJeremie = voitureNicolas; 
4| voitureJeremie.Couleur = "Verte"; 

5 

6 


affichera verte et verte. 


Quoi ? Nous indiquons que la voiture de Nicolas est bleue. Puis nous disons 
que celle de Jérémie est verte et quand on demande d'afficher la couleur des 
deux voitures, on nous dit qu'elles sont vertes toutes les deux alors qu'on 
croyait que celle de Nicolas était bleue ? Tout à l'heure, le fait de changer b 
n'avait pas changé la valeur de a... 


Eh oui, ceci illustre le fait que les classes (comme Voiture) sont des types référence 
et ne possèdent qu’une référence vers une instance de Voiture. Quand nous affectons 
voitureNicolas à voitureJeremie, nous disons en fait que la voiture de Jérémie 
référence la même chose que celle de Nicolas. Concrètement, le C# copie la référence 
de l’objet Voiture qui est contenue dans la variable voitureNicolas dans la variable 
voitureJeremie. Ce sont donc deux variables différentes qui possèdent toutes les deux 
une référence vers l’objet Voiture, qui est la voiture de Nicolas. C'est-à-dire que les 
deux variables référencent le même objet. Ainsi, la modification des propriétés de l’un 
affectera forcément l’autre. 


Inattendu au premier abord, mais finalement, c’est très logique. Comprendre cette 
différence entre les types valeur et les types référence est important, nous verrons 
dans les chapitres suivants quels sont les autres impacts de cette différence. À noter 
également qu’il est impossible de dériver d’un type intégré alors que c’est possible, et 
facile, de dériver d’une classe. 


D'ailleurs, si nous parlions un peu d’héritage ? 


Héritage 
Nous avons vu pour l'instant la théorie de l’héritage. Que les objets chiens héritaient 
des comportements des objets Animaux, que les labradors héritaient des comportements 


des chiens, etc. 
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Passons maintenant à la pratique et créons une classe Animal et une classe Chien qui 
en hérite. Nous allons créer des classes relativement courtes et nous nous limiterons 
dans le nombre d’actions ou de propriétés de celles-ci. Par exemple, nous pourrions 
imaginer que la classe Animal possède une propriété NombreDePattes qui est un entier 
et une méthode Respirer qui affiche le détail de l’action. Ce qui donne : 


1| public class Animal 

2| € 

3 public int NombreDePattes { get; set; } 
4 

5 public void Respirer() 

6 { 

7 Console.WriteLine ("Je respire"); 

8 } 

o|} 


La classe Chien dérive de la classe Animal et peut donc hériter de certains de ses com- 
portements. En l’occurrence, la classe Chien héritera de tout ce qui est public ou pro- 
tégé, identifiés comme vous le savez désormais par les mots-clés public et protected. 
Le chien sait également faire quelque chose qui lui est propre, à savoir aboyer. Il pos- 
sédera donc une méthode supplémentaire. Ce qui donne : 


1| public class Chien : Animal 

2| { 

3 public void Aboyer() 

4 { 

5 Console.WriteLine("Wouaf !"); 
6 } 

7|} 


On représente la notion d’héritage en ajoutant après la classe le caractère « : » suivi de 
la classe mère. Ici, nous avons défini une classe publique Chien qui hérite de la classe 
Animal. 


Nous pouvons dès à présent créer des objets Animal et des objets Chien, par exemple : 


Animal animal = new Animal { NombreDePattes = 4 }; 
animal.Respirer(); 
Console.WriteLine); 


Chien chien = new Chien { NombreDePattes = 4 }; 
chien.Respirer(); 
chien.Aboyer(); 


NO GE À ND 


Si nous exécutons ce code, nous nous rendons bien compte que l’objet Chien, bien que 
n'ayant pas défini la propriété NombreDePattes ou la méthode Respirer() dans le 
corps de sa classe, est capable d’avoir des pattes et de faire l’action respirer : 


Je respire 


Je respire 
Wouaf ! 
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Il a hérité ces comportements de l’objet Animal, en tout cas, ceux qui sont publics. 
Rajoutons deux variables membres de la classe Animal : 


1| public class Animal 

2| € 

3 private bool estVivant; 

4 public int age; 

5 

6 public int NombreDePattes { get; set; } 
T 

8 public void Respirer() 

9 { 

10 Console.WriteLine("Je respire"); 
11 } 

12| } 


L’entier age est public alors que le booléen estVivant est privé. Si nous tentons de les 
utiliser depuis la classe fille Chien, comme ci-dessous : 


1| public class Chien : Animal 

2| € 

3 public void Aboyer() 

4 { 

5 Console.WriteLine("Wouaf !"); 

6 } 

7 

8 public void Vieillir () 

9 { 

10 age++; 

11 } 

12 

13 public void Naissance) 

14 { 

15 age = 0; 

16 estVivant = true; /* Erreur > 
MaPremiereApplication.Animal.estVivant' est 
inaccessible en raison de son niveau de protection 
*/ 

17 } 

18| } 


nous voyons qu'il est tout à fait possible d'utiliser la variable age depuis la méthode 
Vieillir() alors que l’utilisation du booléen estVivant provoque une erreur de com- 
pilation. Vous avez bien compris que celui-ci était inaccessible car il est défini comme 
membre privé. Pour l'utiliser, on pourra le rendre public par exemple. 


Il existe par contre un autre mot-clé qui permet de rendre des variables, propriétés ou 
méthodes inaccessibles depuis un autre objet tout en les rendant accessibles depuis des 
classes filles. Il s’agit du mot-clé protected. Si nous l’utilisons à la place de private 
pour définir la visibilité du booléen estVivant, nous pourrons nous rendre compte que 
la classe Chien peut désormais compiler : 
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1| public class Animal 

2| € 

3 protected bool estVivant; 

4 [... Extrait de code supprimé ...] 
5|} 

6 

7| public class Chien : Animal 

8| 

9 [... Extrait de code supprimé ...] 
10 

11 public void Naissance() 

12 { 

13 age = O0; 

14 estVivant = true; // compilation OK 
15 } 

16| } 


Par contre, cette variable est toujours inaccessible depuis d’autres classes, comme l’est 
également une variable privée. Dans notre classe Program, l'instruction suivante : 


1 | chien.estVivant = true; 
provoquera l'erreur de compilation que désormais nous connaissons bien : 


?MaPremiereApplication.Animal.estVivant ? est inaccessible en 


raison de son niveau de protection 


Le mot-clé protected prend tout son intérêt dès que nous avons à faire avec l'héritage. 
Nous verrons un peu plus loin d’autres exemples de ce mot-clé. 


Nous avons dit dans l'introduction qu’un objet B qui dérive de l’objet À est «une 
sorte » d'objet A. Dans notre exemple du dessus, le Chien est une sorte d’Animal. Cela 
veut dire que nous pouvons utiliser un chien en tant qu'’animal. Par exemple, le code 
suivant : 


1 | Animal animal = new Chien { NombreDePattes = 4 }; 


est tout à fait correct. Nous disons que notre variable animal, de type Animal est une 
instance de Chien. 


Avec cette façon d'écrire, nous avons réellement instancié un objet Chien mais celui- 
ci sera traité en tant qu'Animal. Cela veut dire qu’il sera capable de Respirer() et 
d’avoir des pattes. Par contre, même si en vrai, notre objet est capable d’aboyer, le fait 
qu'il soit manipulé en tant qu’Animal nous empêche de pouvoir le faire Aboyer. Cela 
veut dire que le code suivant : 


1| Animal animal = new Chien { NombreDePattes = 4 }; 
2| animal.Respirer(); 
3| animal.Aboyer(); // erreur de compilation 


provoquera une erreur de compilation pour indiquer que la classe Animal ne contient 
aucune définition pour la méthode Aboyer(). Ce qui est normal, car un animal ne sait 
pas forcément aboyer..…. 
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Quel est l'intérêt alors d'utiliser le chien en tant qu'animal ? 


Bonne question. 


Pour y répondre, nous allons enrichir notre classe Animal, garder notre classe Chien et 
créer une classe Chat qui hérite également d’Animal. Ce pourrait être : 


1| public class Animal 
2| € 
3 protected string prenom; 
4 
5 public void Respirer() 
6 { 
Vé Console.WriteLine("Je suis " + prenom + " et je respire 
"); 
8 } 
o| } 
10 
11| public class Chien : Animal 
12| À 
13 public Chien(string prenomDuChien) 
14 { 
15 prenom = prenomDuChien; 
16 } 
17 
18 public void Aboyer() 
19 { 
20 Console.WriteLine("Wouaf !"); 
21 } 
22| } 
23 
24| public class Chat : Animal 
25 | 
26 public Chat(string prenomDuChat) 
27 { 
28 prenom = prenomDuChat; 
29 } 
30 
31 public void Miauler() 
32 { 
33 Console.WriteLine ("Miaou"); 
34 } 
35| } 


Nous forçons les chiens et les chats à avoir un nom, hérité de la classe Animal, grâce au 
constructeur afin de pouvoir les identifier facilement. Le chat garde le même principe 
que le chien, sauf que nous avons une méthode Miauler() à la place de la méthode 
Aboyer()... Ce qui est, somme toute, logique! L'idée est de pouvoir utiliser nos chiens 
et nos chats ensemble comme des animaux, par exemple en utilisant une liste. 
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Pour illustrer ce fonctionnement, donnons vie à quelques chiens et à quelques chats 
grâce à nos pouvoirs de développeur et mettons-les dans une liste : 


1| List <Animal> animaux = new List<Animal>(); 
2| Animal milou = new Chien("Milou"); 

3| Animal dingo = new Chien("Dingo"); 

4| Animal idefix = new Chien("Idéfix"); 
5| Animal tom = new Chat ("Tom"); 

6| Animal felix = new Chat("Félix"); 

T 

8| animaux.Add(milou); 

9 | animaux.Add(dingo); 

10| animaux.Add(idefix); 

11| animaux.Add(tom) ; 

12| animaux.Add(felix); 


Nous avons dans un premier temps instancié une liste d’animaux à laquelle nous avons 
rajouté 3 chiens et 2 chats, chacun étant considéré comme un animal puisqu'ils sont 
tous des sortes d’animaux, grâce à l’héritage. Maintenant, nous n’avons plus que des 
animaux dans la liste. Il sera donc possible de les faire tous respirer en une simple 
boucle : 


1] foreach (Animal animal in animaux) 
2! { 

3 animal.Respirer (); 

al } 


Ce qui donne : 


Milou_et_je respire 
Dingo et je respire 


Idéfix et je respire 
Tom et je respire 
Félix et je respire 


Et voilà, c’est super simple! 


Imaginez le bonheur de Noé sur son arche quand il a compris que grâce à la POO), il 
pouvait faire respirer tous les animaux en une seule boucle ! Quel travail économisé. 
Peu importe ce qu’il y a dans la liste, des chiens, des chats, des hamsters, nous savons 
que ce sont tous des animaux et qu’ils savent tous respirer. 


Vous avez sans doute remarqué que nous faisons la même chose dans le constructeur 
de la classe Chien et dans celui de la classe Chat. Deux fois la même chose... Ce n’est 
pas terrible. Peut-être y a-t-il un moyen de factoriser tout ça? Effectivement, il est 
possible également d’écrire nos classes de cette façon : 


1| public class Animal 

2| € 

3 protected string prenom; 

4 

5 public Animal(string prenomAnimal) 
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6 { 
7 prenom = prenomAnimal; 
8 } 
9 
10 public void Respirer() 
11 { 
12 Console.WriteLine("Je suis " + prenom + " et je respire 
M) 
13 } 
141} 
15 
16| public class Chien : Animal 
17| 
18 public Chien(string prenomDuChien) : base(prenomDuChien) 
19 { 
20 } 
21 
22 public void Aboyer() 
23 { 
24 Console.WriteLine("Wouaf !"); 
25 } 
26| } 
27 
28| public class Chat : Animal 
29 | 
30 public Chat(string prenomDuChat) : base(prenomDuChat) 
31 { 
32 } 
33 
34 public void Miauler() 
35 { 
36 Console.WriteLine ("Miaou"); 
37 } 
38| } 


Qu'est-ce qui change ? 


Eh bien la classe Animal possède un constructeur qui prend en paramètre un prénom 
et qui le stocke dans sa variable privée. C’est elle qui fait le travail d’initialisation. Il 
devient alors possible pour les constructeurs des classes filles d’appeler le constructeur 
de la classe mère afin de faire l’affectation du prénom. Pour cela, on utilise les deux 
points suivis du mot-clé base qui signifie « appelle-moi le constructeur de la classe 
du dessus » auquel nous passons la variable en paramètre. Avec cette écriture un peu 
barbare, il devient possible de factoriser des initialisations qui ont un sens pour toutes 
les classes filles. Dans notre cas, je veux que tous les objets qui dérivent d’Animal 
puissent facilement définir un prénom. 


Il faut aussi savoir que si nous appelons le constructeur par défaut d’une classe qui 
n’appelle pas explicitement un constructeur spécialisé d’une classe mère, alors celui- 
ci appellera automatiquement le constructeur par défaut de la classe dont il hérite. 
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Modifions à nouveau nos classes pour avoir : 


1| public class Animal 
2| € 
3 protected string prenom; 
4 
5 public Animal () 
6 { 
7 prenom = "Marcel"; 
8 } 
9 
10 public void Respirer() 
11 € 
12 Console.WriteLine("Je suis " + prenom + " et je respire 
DE 
13 } 
1417} 
15 
16| public class Chien : Animal 
17 | 
18 public void Aboyer() 
19 { 
20 Console.WriteLine("Wouaf !"); 
21 } 
22| } 
23 
24| public class Chat : Animal 
25 | { 
26 public Chat(string prenomDuChat) 
27 { 
28 prenom = prenomDuChat; 
29 } 
30 
31 public void Miauler() 
32 { 
33 Console.WriteLine("Miaou"); 
34 } 
35| } 


Ici, la classe Animal met un prénom par défaut dans son constructeur. Le chien n’a pas 
de constructeur et le chat en a un qui accepte un paramètre. 


Il est donc possible de créer un Chien sans qu’il ait de prénom mais il est obligatoire 
d’en définir un pour le chat. Sauf que lorsque nous instancierons notre objet chien, il 
appellera automatiquement le constructeur de la classe mère et tous nos chiens s’ap- 
pelleront Marcel : 


1| static void Main(string[] args) 

2! { 

3 List <Animal> animaux = new List<Animal>(); 
4 Animal chien = new Chien); 

5 Animal tom = new Chat ("Tom"); 
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Animal felix = new Chat ("Félix"); 


animaux.Add(chien); 
animaux.Add(tom); 
animaux.Add(felix); 


foreach (Animal animal in animaux) 
{ 
animal.Respirer(); 


k 


Ce qui affichera : 


Jemsuis Marcel” et je respire 


Je suis Tom et je respire 
Je suis Félix et je respire 


Il est également possible d'appeler un constructeur à partir d’un autre constructeur. 
Prenons l’exemple suivant : 


1 
2 
3 
4 
5 
6 
7 
8 
9 


public class Voiture 


{ 


private int vitesse; 


public Voiture(int vitesseVoiture) 
{ 


vitesse = vitesseVoiture; 


k 


Si nous souhaitons rajouter un constructeur par défaut qui initialise la vitesse à 10 par 
exemple, nous pourrons faire : 


1| public class Voiture 

2| { 

3 private int vitesse; 

4 

5 public Voiture() 

6 { 

ré vitesse = 10; 

8 } 

9 
10 public Voiture(int vitesseVoiture) 
11 { 
12 vitesse = vitesseVoiture; 
13 } 
1|} 

Ou encore : 
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1| public class Voiture 

2| € 

3 private int vitesse; 

4 

5 public Voiture() : this(10) 

6 { 

7 } 

8 

9 public Voiture(int vitesseVoiture) 
10 À 
11 vitesse = vitesseVoiture; 
12 } 
13| } 


Ici, l’utilisation du mot-clé this, suivi d’un entier permet d’appeler le constructeur 
qui possède un paramètre entier au début du constructeur par défaut. Inversement, 
nous pouvons appeler le constructeur par défaut d’une classe depuis un constructeur 
possédant des paramètres afin de pouvoir bénéficier des initialisations de celui-ci : 


1| public class Voiture 

2| € 

3 private int vitesse; 

4 private string couleur; 

5 

6 public Voiture) 

7 { 

8 vitesse = 10; 

9 } 

10 

11 public Voiture(string couleurVoiture) : this() 
12 { 

13 couleur = couleurVoiture; 
14 } 

15| } 


Puisque nous parlons d’héritage, il faut savoir que tous les objets que nous créons ou 
qui sont disponibles dans le framework .NET héritent d’un objet de base. On parle 
en général d’un « super-objet ». L'intérêt de dériver d’un tel objet est de permettre à 
tous les objets d’avoir certains comportements en commun, mais également de pouvoir 
éventuellement tous les traiter en tant qu’objet. Notre super-objet est représenté par 
la classe Object qui définit plusieurs méthodes. Vous les avez déjà vues si vous avez 
regardé dans la complétion automatique après avoir créé un objet. Prenons une classe 
toute vide, par exemple : 


1] public class ObjetVide 
2| { 
3| } 


Si nous instancions cet objet et que nous souhaitons l'utiliser, nous verrons que la 
complétion automatique nous propose des méthodes que nous n’avons jamais créées 
(voir figure 22.1). 
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-inamespace MaPremiereApplication 


{ 
E class Program 
L 
= static void Main(string[] args) 
ObjetVide monObjetVide = new ObjetVide(); 
monObjetVide. 
} © Equals 
} © GetHashCode 
© GetType 


© ToString string object. ToString() 
Retourne un System.String qui représente le System.Object actif. 


FIGURE 22.1 —- La complétion automatique montre des méthodes de la classe de base 
Object 


Nous voyons plusieurs méthodes, comme Equals ou GetHashCode ou GetType ou encore 
ToString. Comme vous l’avez compris, ce sont des méthodes qui sont définies dans la 
classe Object. La méthode ToString par exemple permet d'obtenir une représentation 
de l’objet sous la forme d’une chaîne de caractères. C’est une méthode qui va souvent 
nous servir, nous y reviendrons un peu plus tard. Ce super-objet est du type Object, 
mais on utilise généralement son alias object. 


Aïnsi, il est possible d'utiliser tous nos objets comme des object et ainsi utiliser les 
méthodes qui sont définies sur la classe Object. Ce qui nous permet de faire : 


1| static void Main(string[] args) 

2| { 

3 DbjetVide monObjetVide = new ObjetVide () ; 

4 Chien chien = new Chien); 

5 int age = 30; 

6 string prenom = "Nicolas"; 

7 

8 AfficherRepresentation(monObjetVide); 

9 AfficherRepresentation(chien); 

10 AfficherRepresentation(age); 

11 AfficherRepresentation (prenom); 

12/17} 

13 

14] private static void AfficherRepresentation(object mon0bjetVide) 
15| { 

16 Console.WriteLine (monOUbjetVide.ToString()); 
17] } 


Ce qui affiche : 


MaPremiereApplication.0bjetVide 
MaPremiereApplication.Chien 


30 
Nicolas 
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Comme indiqué, la méthode ToString() permet d'afficher la représentation par défaut 
d’un objet. Vous aurez remarqué qu’il y a une différence suivant ce que nous passons. 
En effet, la représentation par défaut des types référence correspond au nom du type, 
à savoir son espace de nom suivi du nom de sa classe. Pour ce qui est des types valeur, 
la représentation contient en général la valeur du type, à l'exception des structures que 
nous n'avons pas encore vues et que nous aborderons un peu plus loin. 


L'intérêt dans cet exemple de code est de voir que nous pouvons tout manipuler comme 
un object. D'une manière générale, vous aurez peu l’occasion de traiter vos objets en 
tant qu'object car il est vraiment plus intéressant de profiter pleinement du type, 
l’object étant peu utilisable. 


Notez que l'héritage de object est automatique. Nul besoin d'utiliser la 
syntaxe d'héritage que nous avons déjà vue. 


J’en profite maintenant que vous connaissez la méthode ToString() pour parler d’un 
point qui a peut-être titillé vos cerveaux. Dans la première partie, nous avions fait 
quelque chose du genre : 


1| int vitesse = 20; 
2| string chaine = "La vitesse est " + vitesse + " km/h"; 


La variable vitesse est un entier. La chaîne La vitesse est est une chaîne de carac- 
tères. Nous essayons d’ajouter un entier à une chaîne alors que j'ai dit qu’ils n'étaient 
pas compatibles entre eux! Et pourtant cela fonctionne. Effectivement, c’est bizarre! 
Nous concaténons une chaîne à un entier avec l’opérateur + et nous concaténons encore 
une chaîne. 


Et si je fais l’inverse : 


1 | int vitesse = 20 + "40"; 


cela provoque une erreur de compilation. C’est logique, on ne peut pas ajouter un entier 
et une chaîne de caractères. Alors pourquoi cela fonctionne dans l’autre sens ? Ce qui 
se passe en fait dans l'instruction : 


1 | string chaine = "La vitesse est " + vitesse + " km/h"; 


c’est que le compilateur se rend compte que nous concaténons une chaîne avec un 
autre objet, peu importe que ce soit un entier ou un objet complexe. Alors, pour que 
ça fonctionne, il demande une représentation de l’objet sous la forme d’une chaîne de 
caractères. Nous avons vu que ceci se faisait en appelant la méthode ToString() qui 
est héritée de l’objet racine Object. 


L’instruction est donc équivalente à : 


1| string chaine = "La vitesse est " + vitesse.ToString() + " km/h 


LU 
» 
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Dans le cas d’un type valeur comme un entier, la méthode ToString() renvoie la 
représentation interne de la valeur, à savoir « 20 ». Dans le cas d’un objet complexe, 
elle aurait renvoyé le nom du type de l’objet. 


Avant de terminer, il est important d'indiquer que le C;# n'autorise pas l'héritage 
multiple. Ainsi, si nous possédons une classe Carnivore et une classe EtreVivant, il 
ne sera pas possible de faire hériter directement un objet Homme de l’objet Carnivore 
et de l’objet EtreVivant. Ainsi, le code suivant : 


public class Carnivore 
{ 
} 
public class EtreVivant 
{ 
} 


public class Homme : Carnivore, EtreVivant 
{ 
} 


© © I OO Où À CN NN 


+= 
© 


provoquera l'erreur de compilation suivante : 


La classe ?’MaPremiereApplication.Homme” ne peut pas avoir 


plusieurs classes de base : ?’MaPremiereApplication.Carnivore”? 
et ?’EtreVivant”? 


Il est impossible de dériver de deux objets en même temps. 


En revanche, et c’est pertinent, nous pourrons faire un héritage en cascade afin que 
Carnivore dérive de EtreVivant et que Homme dérive de Carnivore : 


public class Carnivore : EtreVivant 
{ 

} 

public class EtreVivant 

{ 

} 


public class Homme : Carnivore 
{ 
} 


© OO 1 OO Où À À ND OH 


+= 
© 


Cependant, il n’est pas toujours pertinent d'opérer de la sorte. Notre Homme pourrait 
être à la fois Carnivore et Frugivore, cependant cela n’a pas de sens qu’un carnivore 
soit également frugivore, ou l’inverse. 
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Tu avais pourtant dit que chaque objet dérivait du super-objet Object, mais 
s'il dérive d'une autre classe comme un chien dérive d'un animal, ça fait bien 
deux classes dont il dérive... 


Effectivement, mais dans ce cas-là, ce n’est pas pareil. Comme il est automatique de 
dériver de object, c’est comme si on avait le chien qui hérite de animal qui hérite 
lui-même de object. Le C## est assez malin pour ça! 


Substitution 


Nous avons vu juste avant l’utilisation de la méthode ToString() qui permet d'obtenir 
la représentation d’un objet sous forme de chaîne de caractères. En l’occurrence, vous 
conviendrez avec moi que la représentation de notre classe Chien n’est pas particuliè- 
rement exploitable. Le nom du type c’est bien, mais ce n’est pas très parlant. 


Ca serait pas mal que, quand nous demandons d’afficher un chien, nous obtenions 
le nom du chien, vous ne trouvez pas? C’est là qu'intervient la substitution. Nous 
en avons parlé dans l'introduction à la POO, la substitution permet de redéfinir un 
comportement dont l’objet a hérité afin qu’il corresponde aux besoins de l’objet fils. 
Typiquement, ici, la méthode ToString() du super-objet ne nous convient pas et dans 
le cas de notre chien, nous souhaitons la redéfinir, en écrire une nouvelle version. 


Pour cet exemple, simplifions notre classe Chien afin qu’elle n’ait qu’une propriété pour 
stocker son prénom : 


1| public class Chien 

2| € 

3 public string Prenom { get; set; } 
al} 


Pour redéfinir la méthode ToString() nous allons devoir utiliser le mot-clé override 
qui signifie que nous souhaitons substituer la méthode existante afin de remplacer son 
comportement, ce que nous pourrons écrire en C# avec : 


1| public class Chien 

2| € 

3 public string Prenom { get; set; } 

4 

5 public override string ToString() 

6 { 

7 return "Je suis un chien et je m'appelle " + Prenom; 
8 } 

o[} 


Le mot-clé override se met avant le type de retour de la méthode, comme on peut le 
voir ci-dessus. Si nous appelons désormais la méthode ToString de notre objet Chien : 


1| Chien chien = new Chien { Prenom = "Max" }; 
2| Console.WriteLine(chien.ToString()); 
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notre programme va utiliser la nouvelle version de la méthode ToString() : 


Je suis un chien et je m’appelle Max 


Et voilà un bon moyen d'utiliser la substitution, la représentation de notre objet est 
quand même plus parlante! Adaptons désormais cet exemple à nos classes. Pour mon- 
trer comment faire, reprenons notre classe Chien qui possède une méthode Aboyer () : 


1| public class Chien 

2| { 

3 public void Aboyer() 

4 { 

5 Console.WriteLine("Wouaf !"); 
6 } 

71} 


Nous pourrions imaginer de créer une classe ChienMuet qui dérive de la classe Chien 
et qui hérite donc de ses comportements. Mais, que penser d’un chien muet qui serait 
capable d’aboyer ? Cela n’a pas de sens! Il faut donc redéfinir cette fichue méthode. 


Utilisons alors le mot-clé override comme nous l’avons vu pour obtenir : 


1| public class ChienMuet : Chien 

2| € 

3 public override void Aboyer() 
4 { 

5 Console.WriteLine("..."); 
6 } 

7|} 


Créons un chien muet puis faisons-le aboyer, cela donne : 


1| ChienMuet pauvreChien = new ChienMuet(); 
2| pauvreChien.Aboyer (); 


Sauf que nous rencontrons un problème. Si nous tentons de compiler ce code, Visual 
C+# Express nous génère une erreur de compilation : 


MaPremiereApplication.Program.ChienMuet.Aboyer()? : ne peut pas 
substituer le membre hérité ?’MaPremiereApplication.Program. 


Chien.Aboyer()?, car il n’est pas marqué comme virtual, 
abstract ou override. 


En réalité, pour pouvoir créer une méthode qui remplace une autre, il faut qu’une condi- 
tion supplémentaire soit vérifiée : il faut que la méthode à remplacer s’annonce 
comme candidate à la substitution. Cela veut dire que l’on ne peut pas substituer 
n'importe quelle méthode, mais seulement celles qui acceptent de l'être. C’est le cas 
pour la méthode ToString que nous avons vue précédemment. Les concepteurs du fra- 
mework .NET ont autorisé cette éventualité. Heureusement, sinon, nous serions bien 
embêtés ! 
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Pour marquer notre méthode Aboyer de la classe Chien comme candidate éventuelle à 
la substitution, il faut la préfixer du mot-clé virtual. Ainsi, elle annonce à ses futures 
filles que si elles le souhaitent, elles peuvent redéfinir cette méthode. 


Cela se traduit ainsi dans le code : 


1| public class Chien 

2| { 

3 public virtual void Aboyer() 
4 { 

5 Console.WriteLine("Wouaf !"); 
6 } 

7|} 

8 

9| public class ChienMuet : Chien 

10! 

11 public override void Aboyer() 
12 { 
13 Console.WriteLine("..."); 
14 } 
15| } 


Désormais, l’instanciation de l’objet est possible et nous pourrons avoir notre code : 


1 
2 


ChienMuet pauvreChien = new ChienMuet(); 
pauvreChien.Aboyer(); 


Ce code affichera : 


En | 


Parfait ! Tout est rentré dans l’ordre. 


Le message d'erreur, quoique peu explicite, nous mettait quand même sur la bonne 
voie. Visual C# Express nous disait qu’il fallait que la méthode soit marquée comme 
virtual, ce que nous avons fait. Il proposait également qu’elle soit marquée abstract, 
nous verrons un peu plus loin ce que ça veut dire. Visual C# Express indiquait enfin 
que la méthode pouvait être marquée override. Cela veut dire qu’une classe fille 
de ChienMuet peut également redéfinir la méthode Aboyer() afin qu’elle colle à ses 
besoins. Elle n’est pas marquée virtual mais elle est marquée override. Par exemple : 


1| public class ChienMuetAvecSyntheseVocale : ChienMuet 
2| € 

3 public override void Aboyer() 

4 { 

5 Console.WriteLine("bwarf !"); 

6 } 

7|} 


Il y à encore un dernier point que nous n’avons pas abordé. Il s’agit de la capacité 
pour une classe fille de redéfinir une méthode tout en conservant la fonctionnalité de 
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la méthode de la classe mère. Imaginons notre classe Animal qui possède une méthode 
Manger () : 


1| public class Animal 

2| { 

3 public virtual void Manger () 

4 { 

5 Console.WriteLine("Mettre les aliments dans la bouche") 
6 Console.WriteLine ("Mastiquer"); 

7 Console.WriteLine("Avaler"); 

8 Console.WriteLine("..."); 

9 } 

10| } 


Notre classe Chien pourra s'appuyer sur le comportement de la méthode Manger () 
de la classe Animal pour créer sa propre action personnelle. Cela se passe en utilisant 
à nouveau le mot-clé base qui représente la classe mère. Nous pourrons par exemple 
appeler la méthode Manger () de la classe mère afin de réutiliser son fonctionnement. 
Cela donne : 


1| public class Chien : Animal 

2| € 

3 public override void Manger () 

4 { 

5 Console.WriteLine('"Réclamer à manger au maître"); 
6 base.Manger () ; 

7 Console.WriteLine("Remuer la queue"); 

8 } 

9| } 


Dans cet exemple, je fais quelque chose avant d’appeler la méthode de la classe mère, 
puis je fais quelque chose d’autre après. Maintenant, si nous faisons manger notre chien : 


1] Chien chien = new Chien); 
2| chien.Manger(); 


s’affichera dans la console toute la série des actions définies comme étant « manger » : 


Réclamer à manger au maître 

Mettre les aliments dans la bouche 
Mastiquer 

Avaler 


Remuer la queue 


Nous voyons bien avec cet exemple comment la classe fille peut réutiliser les méthodes 
de sa classe mère. 


À noter qu'on peut également parler de spécialisation ou de redéfinition à 
la place de la substitution. 
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Polymorphisme 


Nous avons dit qu’une manifestation du polymorphisme était la capacité pour une classe 
d'effectuer la même action sur différents types d'intervenants. Il s’agit de la surcharge, 
appelée aussi polymorphisme ad hoc. 


Concrètement, cela veut dire qu’il est possible de définir la même méthode avec des 
paramètres en entrée différents. Si vous vous rappelez bien, c’est quelque chose que nous 
avons déjà fait sans le savoir. Devinez ... Oui, c’est ça, avec la méthode 
Console.WriteLine. 


Nous avons pu afficher des chaînes de caractères, mais aussi des entiers, même des types 
double, et plus récemment des objets. Comment ceci est possible alors que nous avons 
déjà vu qu’il était impossible de passer des types en paramètres d’une méthode qui ne 
correspondent pas à sa signature ? ! 


Ainsi, l'exemple suivant : 


1] public class Program 

| € 

3 static void Main(string[]l args) 

4 { 

5 Math math = new Math(); 

6 int a = 5; 

7 int b = 6; 

8 int resultat = math.Addition(a, b); 

9 

10 double c = 1.5; 

11 double d = 5.0; 

12 resultat = math.Addition(c, d); /* erreur de 
compilation */ 

13 } 

11} 

15 

16| public class Math 

17 | { 

18 public int Addition(int a, int b) 

19 { 

20 return à + b; 

21 he 

52,11 


provoquera une erreur de compilation lorsque nous allons essayer de passer des variables 
du type double à notre méthode qui prend des entiers en paramètres. Pour que ceci 
fonctionne, nous allons rendre polymorphe cette méthode en définissant à nouveau 
cette même méthode mais en lui faisant prendre des paramètres d’entrée différents : 


1] public class Program 

2| { 

3 static void Main(string[l args) 
4 { 

5 Math math = new Math(); 
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6 int a = 5; 

7 int b = 6; 

8 int resultat = math.Addition(a, bj); 

9 

10 double c = 1.5; 

11 double d = 5.0; 

12 double resultatDouble = math.Addition(c, d); /* ça 
compile, youpi */ 

13 } 

14! } 

15 

16| public class Math 

17 | À 

18 public int Addition(int a, int b) 

19 À 

20 return a + b; 

21 } 

22 

23 public double Addition(double a, double b) 

24 { 

25 return à + b; 

26 F 

27| } 


Nous avons ainsi écrit deux formes différentes de la même méthode. Une qui accepte des 
entiers et l’autre qui accepte des double. Ce code fonctionne désormais correctement. 


Il est bien sûr possible d'écrire cette méthode avec beaucoup de paramètres de types 
différents, même une classe Chien, en imaginant que le fait d’additionner deux chiens 
correspond au fait d’additionner leurs nombres de pattes : 


1| public class Math 

2| { 

3 public int Addition(int a, int b) 

4 { 

5 return à + b; 

6 } 

7 

8 public double Addition(double a, double b) 
9 { 

10 return a + b; 

11 } 

12 

13 public int Addition(Chien ci, Chien c2) 

14 { 

15 return ci.NombreDePattes + c2.NombreDePattes; 
16 } 

17 | } 


Attention, j'ai toujours indiqué qu’il était possible d’ajouter une nouvelle forme à la 
même méthode en changeant les paramètres d’entrées. Vous ne pourrez pas le faire en 
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changeant uniquement le paramètre de retour. Ce qui fait que cet exemple ne pourra 
pas compiler : 


1| public class Math 

21 { 

3 public int Addition(int a, int b) 
4 { 

5 return à + b; 

6 F 

7 

8 public double Addition(int a, int b) 
9 { 

10 return à + b; 

11 } 

12| +} 


Les deux méthodes acceptent deux entiers en paramètres et renvoient soit un entier, 
soit un double. Le compilateur ne sera pas capable de choisir quelle méthode utiliser 
lorsque nous essayerons d’appeler cette méthode. Les méthodes doivent se différencier 
avec les paramètres d’entrées. Lorsque nous avons plusieurs signatures possibles pour 
la même méthode, vous remarquerez que la complétion automatique nous propose alors 
plusieurs possibilités (voir figure 22.2). 


MaPremiereApplication.Program + | 3% Main(strir 


E static void Main(string[] args) 
{ 
Math math = new Math(); 
inta =5; 
int b = 6; 
int resultat = math.Addition(a, b); 


double ç = 1.5; 
double d = 5.08; 
double resultatDouble ô ate je ë 


+ A 1sur3 w int Math.Addition(Chien c1, Chien c2) | 


FIGURE 22.2 —- La complétion automatique propose plusieurs signatures de la même 
méthode 


Visual C# indique qu’il a trois méthodes possibles qui s'appellent Math.Addition. Pour 
voir la signature des autres méthodes, il suffit de cliquer sur les flèches, ou d'utiliser les 
flèches du clavier, comme indiqué à la figure 22.3. 


C’est ce qui se passe dans la méthode Console .WriteLine (voir figure 22.4). 


Nous voyons ici qu'il existe 19 écritures de la méthode WriteLine, la cinquième prenant 
en paramètre un décimal. Notez que pour écrire plusieurs formes de cette méthode, nous 
pouvons également jouer sur le nombre de paramètres. La méthode : 


1| public int Addition(int a, int b, int c) 
2! { 

3 return à + b + c; 

al } 
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static void Main(string[] args) 


{ 


Math math = new Math(); 

int a=5; 

int b =6; 

int resultat = math.Addition(a, b); 


double ç = 1.5; 
double d = 5.0; 
double resultatDouble = math.Addition( 


L A 2sur3 w double Math.Addition(double a, double b) | 


FIGURE 22.3 - La complétion automatique présente plusieurs signatures de la méthode 
Addition 


#$ MaPremiereApplication.Program 
E static void Main(string[] args) 


{ 


3% Main(string[] args) 


Console.WriteLine( 
LE. [a 5sur19 w void Console. WriteLine(decimal value) 
Écrit dans le flux de sortie standard la représentation textuelle de la valeur System.Decimal spécifiée, suivie du terminateur de la ligne active. 
value: Valeur à écrire. 


} 


FIGURE 22.4 - La complétion automatique présente plusieurs signatures de la méthode 
Console.WriteLine 


sera bien une nouvelle forme de la méthode Addition. 


Nous avons également vu dans le chapitre sur les constructeurs d’une classe qu’il était 
possible de cumuler les constructeurs avec des paramètres différents. Il s’agit à nouveau 
du polymorphisme. Il nous permet de définir différents constructeurs sur nos objets. 


La conversion entre les objets avec le casting 


Nous avons déjà vu dans la partie précédente qu’il était possible de convertir les types 
qui se ressemblent entre eux. Cela fonctionne également avec les objets. Plus précisé- 
ment, cela veut dire que nous pouvons convertir un objet en un autre seulement s’il est 
une sorte de l’autre objet. Nous avons vu dans les chapitres précédents qu’il s'agissait 
de la notion d’héritage. 


Ainsi, si nous avons défini une classe Animal et que nous définissons une classe Chien 
qui hérite de cette classe Animal : 


public class Animal 
{ 
} 


public class Chien : Animal 


{ 


1 
2 
3 
4 
5 
6 
7|} 


nous pourrons alors convertir le chien en animal dans la mesure où le chien est une 
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sorte d’animal : 


1| Chien medor = new Chien); 
2| Animal animal = (Animal)medor; 


Nous utilisons pour ce faire un cast, comme nous l’avons déjà fait pour les types intégrés 
(int, bool, etc.). Il suffit de préfixer la variable à convertir du type entre parenthèses 
dans lequel nous souhaitons le convertir. Ici, nous pouvons convertir facilement notre 
Chien en Animal. 


Par contre, il est impossible de convertir un chien en voiture, car il n’y à pas de relation 
d’héritage entre les deux. Ainsi les instructions suivantes : 


1| Chien medor = new Chien); 
2| Voiture voiture = (Voiture)medor; 


provoqueront une erreur de compilation. Nous avons précédemment utilisé l’héritage 
afin de mettre des chiens et des chats dans une liste d’animaux. Nous avions fait quelque 
chose du genre : 


List <Animal> animaux = new List<Animal>(); 
Animal chien = new Chien); 
Animal chat = new Chat(); 


animaux.Add(chien); 
animaux.Add(chat); 


Oo Où À & ND 


Il serait plus logique en fait d’écrire les instructions suivantes : 


List <Animal> animaux = new List<Animal>(); 
Chien chien = new Chien); 
Chat chat = new Chat); 


animaux.Add((Animal)chien); 
animaux.Add((Animal)chat); 


OO Où & & ND + 


Dans ce cas, nous créons un objet Chien et un objet Chat que nous mettons dans une 
liste d'objets Animal grâce à une conversion utilisant un cast. En fait, ce cast est inutile 
et nous pouvons simplement écrire : 


List <Animal> animaux = new List<Animal>(); 
Chien chien = new Chien); 
Chat chat = new Chat); 


animaux.Add(chien); 
animaux.Add(chat); 


OO Où & & ND + 


La conversion est implicite, comme lorsque nous avions utilisé un object en paramètres 
d’une méthode et que nous pouvions lui passer tous les types qui dérivent d’object. 
Nous avions également vu que nous pouvions traiter les chiens et les chats comme 
des animaux à partir du moment où nous les mettions dans une liste. Avec les objets 
suivants : 
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public class Animal 


né 
public void Respirer() 
{ 
Console.WriteLine ("Je respire"); 
} 
} 
public class Chien : Animal 
{ 
public void Aboyer() 
{ 
Console.WriteLine ("Waouf"); 
} 
} 
public class Chat : Animal 
{ 
public void Miauler() 
{ 
Console.WriteLine ("Miaou"); 
} 
} 


Nous pouvions utiliser une boucle pour faire respirer tous nos animaux : 


© ® I OO OÙ À À NN H 


= 
hR © 


List<Animal> animaux = new List<Animal>(); 
Chien chien = new Chien); 
Chat chat = new Chat(); 


animaux.Add(chien); 
animaux.Add(chat); 


foreach (Animal animal in animaux) 
{ 


animal.Respirer(); 


Mais impossible de faire aboyer le chien, ni miauler le chat. Si vous tentez de remplacer 
dans la boucle Animal par Chien, avec : 


1 
2 
3 
4 


foreach (Chien c in animaux) 
{ 
c.Aboyer (); 


Vous pourrez faire aboyer le premier élément de la liste qui est effectivement un chien, 


par 


contre il y aura un plantage au deuxième élément de la liste car il s’agit d’un chat : 


Waouf 
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Exception non gérée : System.InvalidCastException: Impossible d’? 


effectuer un cast d’un objet de type ?’MaPremiereApplication. 
Chat” en type ?’MaPremiereApplication.Chien’. 

à MaPremiereApplication.Program.Main(String[] args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
19 


Lorsque notre programme à tenté de convertir un animal qui est un chat en chien, il 
nous à fait comprendre qu’il n’appréciait que moyennement. Les chiens n’aiment pas 
trop les chats d’une manière générale, alors en plus, un chat qui essaie de se faire passer 
pour un chien : c’est une déclaration de guerre ! Voilà pourquoi notre programme a levé 
une exception. Il lui était impossible de convertir un Chat en Chien. 


Il est cependant possible de tester si une variable correspond à un objet grâce au mot- 
clé is. Ce qui nous permettra de faire la conversion adéquate et de nous éviter une 
erreur à l'exécution : 


1] foreach (Animal animal in animaux) 
2! À 

3 if (animal is Chien) 

4 { 

5 Chien c = (Chien)animal; 
6 c.Aboyer(); 

7 } 

8 if (animal is Chat) 

9 { 

10 Chat c = (Chat)animal; 
11 c.Miauler(); 

12 } 

131 } 


Nous testons avec le mot-clé is si l’animal est une instance d’un chien ou d’un chat. 
Le code du dessus nous permettra d'utiliser dans la boucle l’animal courant comme un 
chien ou un chat en fonction de ce qu’il est vraiment, grâce au test : 


Waouf 
Miaou 


Le fait de tester ce qu'est vraiment l’animal avant de le convertir est une sécurité 
indispensable pour éviter ce genre d’erreur. C’est l’inconvénient du cast explicite. Il 
convient très bien si nous sommes certains du type dans lequel nous souhaitons en 
convertir un autre. Par contre, si la conversion n’est pas possible, alors nous aurons 
une erreur. Lorsque nous ne sommes pas certains du résultat du cast, mieux vaut tester 
si l'instance d’un objet correspond bien à l’objet lui-même. 


Cela peut se faire comme nous l’avons vu avec le mot-clé is, mais également avec un 
autre cast qui s'appelle le cast dynamique. Il se fait en employant le mot-clé as. Ce 
cast dynamique vérifie que l’objet est bien convertible. Si c’est le cas, alors il fait un 
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cast explicite pour renvoyer le résultat de la conversion, sinon, il renvoie une référence 
nulle. Le code du dessus peut donc s’écrire : 


1] foreach (Animal animal in animaux) 
2! À 

a Chien ci = animal as Chien; 
4 if (cl = null) 

5 € 

6 ci.Aboyer(); 

7 } 

8 Chat c2 = animal as Chat; 

9 if (c2 != null) 

10 { 

11 c2.Miauler(); 

12 } 

13| } 


On utilise le mot-clé as en le faisant précéder de la valeur à tenter de convertir et en 
le faisant suivre du type dans lequel nous souhaitons la convertir. 


Fonctionnellement, nous faisons la même chose dans les deux codes. Vous pouvez choisir 
l'écriture que vous préférez, mais sachez que c’est ce dernier qui est en général utilisé, 
car il est préconisé par Microsoft. De plus, il est un tout petit peu plus performant. 


Un petit détail encore. Il est possible de convertir un type valeur, comme un int ou 
un string en type référence en utilisant ce qu’on appelle le boxing. Rien à voir avec 
le fait de taper sur les types valeur! Comme nous l’avons vu, les types valeur et les 
types référence sont gérés différemment par .NET. Aussi, si nous convertissons un type 
valeur en type référence, .NET fait une opération spéciale automatiquement. Aïnsi le 
code suivant : 


1| int i = 5; 
2| object o = i; // boxing 


effectue un boxing automatique de l’entier en type référence. C’est ce boxing automa- 
tique qui nous permet de manipuler les types valeur comme des object. C’est aussi 
ce qui nous à permis plus haut de passer un entier en paramètre à une méthode qui 
acceptait un object. En interne, ce qui se passe c’est que object se voit attribuer une 
référence vers une copie de la valeur de i. Ainsi, modifier o ne modifiera pas i. Ce code : 
int i = 5; 

object o = i; // boxing 

o = 6; 

Console.WriteLine(i); 

Console.WriteLine(o); 


où à À DD 


affiche 5 puis 6. Le contraire est également possible, ce qu’on appelle l’unboxing. 
Seulement, celui-ci a besoin d’un cast explicite afin de pouvoir compiler. C'est-à-dire : 


1| int À = 5; 
2| object o = i; // boxing 
3| int j = (int)o; // unboxing 
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ici nous reconvertissons la référence vers la valeur de o en entier et nous effectuons à 
nouveau une copie de cette valeur pour la mettre dans j. Ainsi le code suivant : 


int À = 5; 

object o = i; // boxing 

o = 6; 

int j = (int)o; // unboxing 
J = 7; 


Console.WriteLine(i); 
Console.WriteLine(o); 
Console.WriteLine(j); 


© © I OO OÙ À À NN 


affichera en toute logique 5 puis 6 puis 7. 


À noter que ces opérations sont consommatrices de temps, elles sont donc à 
faire le moins possible. 


En résumé 


— Les objets peuvent être des types valeur ou des types référence. Les variables de 
type valeur possèdent la valeur de l’objet, comme un entier. Les variables de type 
référence possèdent une référence vers l’objet en mémoire. 

— Tous les objets dérivent de la classe de base Object. 

— On peut substituer une méthode grâce au mot-clé override si elle s’est déclarée 
candidate à la substitution grâce au mot-clé virtual. 

— La surcharge est le polymorphisme permettant de faire varier les types des paramètres 
d’une même méthode ou leurs nombres. 

— Il est possible grâce au cast de convertir un type en un autre type, s’ils ont une 
relation d’héritage. 
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Lim 23 | 


Notions avancées de POO en CÆ 


Difficulté : 


ans ce chapitre, nous allons continuer à découvrir comment nous pouvons faire de 
l'orienté objet avec le CH. Nous allons pousser un peu plus loin en découvrant les 
interfaces et en manipulant les classes statiques et abstraites. 


À la fin de ce chapitre, vous serez capables de faire des objets encore plus évolués et vous 
devriez être capables de créer un vrai petit programme orienté objet... | 
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Comparer des objets 


Nous avons vu dans la première partie qu’il était possible de comparer facilement des 
types valeur grâce aux opérateurs de comparaison. En effet, vu que des variables de 
ces types possèdent directement la valeur que nous lui affectons, on peut facilement 
comparer un entier avec la valeur 5, ou un entier avec un autre entier. Par contre, 
cela ne fonctionne pas avec les objets. En effet, nous avons vu que les variables qui 
représentent des instances d’objet contiennent en fait une référence vers l’instance. 


Cela n’a pas vraiment de sens de comparer des références. De plus, en imaginant que je 
veuille vraiment comparer deux voitures, sur quels critères puis-je déterminer qu’elles 
sont égales ? La couleur ? La vitesse ? 


Sans rien faire, la comparaison en utilisant par exemple l'opérateur d'égalité « —— » 
permet simplement de vérifier si les références pointent vers le même objet. 


Pour les exemples de ce chapitre, nous nous baserons sur la classe Voiture suivante : 


1| public class Voiture 

à |! 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 

6| } 


Ainsi, si nous écrivons : 


1| Voiture voitureNicolas = new Voiture); 

2| voitureNicolas.Couleur = "Bleue; 

3| Voiture voitureJeremie = voitureNicolas; 

4| voitureJeremie.Couleur = "Verte"; 

5| if (voitureJeremie == voitureNicolas) 

6| { 

7 Console.WriteLine("Les objets référencent la même instance" 
); 

8| } 


Ce code affichera la chaîne « Les objets référencent la même instance » car effectivement, 
nous avons affecté la référence de voitureNicolas à voitureJeremie - ce qui implique 
également que la modification de la voiture de Jérémie affecte également la voiture de 
Nicolas, comme nous l’avons déjà vu. 


Par contre, le code suivant : 


1| Voiture voitureNicolas = new Voiture); 

2| Voiture voitureJeremie = new Voiture); 

3| if (voitureJeremie == voitureNicolas) 

al { 

5 Console.WriteLine("Les objets référencent la même instance" 
DE 

6| } 


n’affichera évidemment rien car ce sont deux instances différentes. 
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S'il s'avère qu'il est vraiment pertinent de comparer deux voitures entre elles, il faut 
savoir que c’est quand même possible. La première chose à faire est de définir les critères 
de comparaison. Par exemple, nous n’avons qu’à dire que deux voitures sont identiques 
quand la couleur, la marque et la vitesse sont égales. Je sais, c’est un peu irréel, mais 
c’est pour l’exemple. 


Ainsi, nous pourrons par exemple vérifier que deux voitures sont égales avec l’instruc- 
tion suivante : 


1| if (voitureNicolas.Couleur == voitureJeremie.Couleur && 
voitureNicolas.Marque == voitureJeremie.Marque && 
voitureNicolas.Vitesse == voitureJeremie.Vitesse) 
2| { 
3 Console.WriteLine("Les deux voitures sont identiques"); 
} 


La comparaison d'égalité entre deux objets, c’est en fait le rôle de la méthode Equals () 
dont chaque objet hérite de la classe mère Object. À part pour les types valeur, le com- 
portement par défaut de la méthode Equals() est de comparer les références des objets. 
Seulement, il est possible de définir un comportement plus approprié pour notre classe 
Voiture, grâce à la fameuse spécialisation. Pour plus d’informations sur la méthode 
Equals(), je vous renvoie au code web suivant : 


Méthode Equals 
Code web : 657987 


Comme on l’a déjà vu, on utilise le mot-clé override. Ceci est possible dans la me- 
sure où la classe Object a défini la méthode Equals comme virtuelle, avec le mot-clé 
virtual. Ce qui donne : 


1| public class Voiture 

2| € 

3 public string Couleur { get; set; } 

4 public string Marque { get; set; } 

5 public int Vitesse { get; set; } 

6 

7 public override bool Equals(object obj) 

8 { 

9 Voiture v = obj as Voiture; 

10 if (v == null) 

11 return false; 

12 return Vitesse == v.Vitesse && Couleur == v.Couleur && 
Marque == v.Marque; 

13 } 

1417} 


Remarquons que la méthode Equals prend en paramètre un object. La première chose 
à faire est donc de vérifier que nous avons réellement une voiture, grâce au cast dy- 
namique. Ensuite, il ne reste qu’à comparer les propriétés de l’instance courante et de 
l’objet passé en paramètre. 


Pour faire une comparaison entre deux voitures, nous pourrons utiliser le code suivant : 
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1| Voiture voitureNicolas = new Voiture { Vitesse = 10, Marque = " 
Peugeot", Couleur = "Grise"}; 

2| Voiture voitureJeremie = new Voiture { Vitesse = 10, Marque = " 
Peugeot", Couleur = "Grise" }; 

3| if (voitureNicolas.Equals(voitureJeremie)) 

alt 

5 Console.WriteLine("Les objets ont les mêmes valeurs dans 


leurs propriétés"); 


6| } 


Nos deux voitures sont identiques car leurs marques, leurs couleurs et leurs vitesses 
sont identiques : 


Les objets ont les mêmes valeurs dans leurs propriétés 


C’est facile de comparer ! Sauf que vous aurez peut-être remarqué que la compilation 
de ce code provoque un avertissement. Il ne s’agit pas d’une erreur, mais Visual C# 
Express nous informe qu’il faut faire attention (voir la figure 23.1). 


Liste d'erreurs 


|@ 0 erreurs || AA 1 avertissement || (1) 0 messages 
| Description . Fichier 
& 1 ‘MaPremiereApplication.Voiture se substitue à Object.Equals(object o) mais pas à Object.GetHashCode() Voiture.cs 


FIGURE 23.1 —- La compilation fait apparaître un avertissement 


Il nous dit que nous avons substitué la méthode Equals () sans avoir redéfini la méthode 
GetHashCode(). Nous n’avons pas besoin ici de savoir à quoi sert vraiment la méthode 
GetHashCode(), mais si l’envie vous prend de vous instruire à ce sujet, vous pouvez 
lire la documentation officielle disponible via le code web suivant : 


Méthode GetHashCode() 
Code web : 785089 


Toujours est-il que nous devons rajouter une spécialisation de la méthode GetHashCode(), 
dont le but est de renvoyer un identifiant plus ou moins unique représentant l’objet, ce 
qui donnera : 


1| public class Voiture 

2| € 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 
6 

7 

8 

9 


public override bool Equals(object obj) 
{ 


Voiture v = obj as Voiture; 
10 if (v == null) 
ii return false; 
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12 return Vitesse == v.Vitesse && Couleur == v.Couleur && 
Marque == v.Marque; 

13 } 

14 

15 public override int GetHashCode() 

16 { 

17 return Couleur.GetHashCode() * Marque.GetHashCode() * 
Vitesse.GetHashCode); 

18 k 

19! } 


Nous nous servons du fait que chaque variable de la classe possède déjà un identifiant 
obtenu avec la méthode GetHashCode(). En combinant chaque identifiant de chaque 
propriété nous pouvons en créer un nouveau. Ici, la classe est complète et prête à 
être comparée. Elle pourra donc fonctionner correctement avec tous les algorithmes 
d'égalité du framework .NET, comme les clés de hachage. Notez quand même que 
devoir substituer ces deux méthodes est une opération relativement rare. 


Ce qu'il est important de retenir, c’est ce fameux warning. La conclusion à tirer est que 
notre façon de comparer, bien que fonctionnelle pour notre voiture, n’est pas parfaite. 


Pourquoi? Parce qu’en ayant substitué la méthode Equals(), nous croyons que la 
comparaison est bonne sauf que le compilateur nous apprend que ce n’est pas le cas. 
Heureusement qu'il était là, ce compilateur ! Comme c’est une erreur classique, il est 
capable de la détecter. Mais si c’est autre chose et qu’il ne le détecte pas ? 


Tout ça manque d’uniformisation, vous ne trouvez pas ? Il faudrait quelque chose qui 
nous assure que la classe est correctement comparable. Une espèce de contrat que l’objet 
s’engagerait à respecter pour être sûr que toutes les comparaisons soient valides. 


Un contrat? Un truc qui finit par « able » ? Ça me rappelle quelque chose ça... mais 
oui, les interfaces ! 


Les interfaces 


Une fois n’est pas coutume. Plutôt que de commencer par étudier le plus simple, nous 
allons étudier le plus logique puis nous reviendrons sur le plus simple. C'est-à-dire que 
nous allons pousser un peu plus loin la comparaison en nous servant des interfaces et 
nous reviendrons ensuite sur le moyen de créer une interface. 


Nous avons donc dit qu’une interface était un contrat que s’engageait à respecter un 
objet. C’est tout à fait ce dont on a besoin ici. Notre objet doit s'engager à fonctionner 
pour tous les types de comparaison. Il doit être comparable. Mais comparable ne veut 
pas forcément dire « égalité », nous devrions être aussi capables d’indiquer si un objet 
est supérieur à un autre. 


Pourquoi ? Imaginons que nous possédions un tableau de voitures et que nous souhai- 
tions le trier comme on à vu dans un chapitre précédent. Pour les entiers c'était une 
opération plutôt simple ; avec la méthode Array.Sort() ils étaient automatiquement 
triés par ordre croissant. Mais dans notre cas, comment un tableau sera capable de 
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trier nos voitures ? 


Nous voici donc en présence d’un cas concret d'utilisation des interfaces. L'interface 
IComparable permet de définir un contrat de méthodes destinées à la prise en charge 
de la comparaison entre deux instances d’un objet. Pour la documentation sur cette 
méthode, je vous renvoie au code web suivant : 


u eee 
Code web : 802146 
Une fois ces méthodes implémentées, nous serons certains que nos objets seront com- 


parables correctement. Pour cela, nous allons faire en sorte que notre classe « implé- 
mente » l'interface. 


Reprenons notre classe Voiture avec uniquement ses propriétés et faisons lui implé- 
menter l'interface IComparable. Pour implémenter une interface, on utilisera la même 
syntaxe que pour hériter d’une classe, c’est-à-dire qu’on utilisera les deux points suivis 
du type de l'interface. Ce qui donne : 


1| public class Voiture : IComparable 

2 |. 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 

6 


Il est conventionnel de démarrer le nom d'une interface par un | majuscule. 
C'est le cas de toutes les interfaces du framework .NET. 


Si vous tentez de compiler ce code, vous aurez un message d'erreur (voir figure 23.2). 


[@ 1 ereur JA 0 avertissements |] 0 message | 


Description = | F 


@1 ‘MaPremiereApplication.Voiture' n'implémente pas le membre d'interface ‘System.IComparable.CompareTo(object)' Vi 


FIGURE 23.2 — Erreur de compilation car l'interface n’est pas implémentée 


Le compilateur nous rappelle à l’ordre : nous annonçons que nous souhaitons respecter 
le contrat de comparaison, sauf que nous n’avons pas la méthode adéquate ! 


Eh oui, le contrat indique ce que nous nous engageons à faire mais pas la façon de le 
faire. L’implémentation de la méthode est à notre charge. 


Toujours dans l’optique de simplifier la tâche du développeur, Visual C# Express 
nous aide pour implémenter les méthodes d’un contrat. Faites un clic droit sur l’inter- 
face et choisissez dans le menu contextuel Implémenter l’interface et Implémenter 
l’interface (voir figure 23.3). 


226 


LES INTERFACES 


=inamespace MaPremiereApplication 


= public class Voiture : IComparaf— 


f Implémenter l'interface »| Implémenter l'interface ] 
publie string Couleur (BR Refactoriser C)) Implémenter l'interface explicitement 
public string Marque { get; À 2 : L 
public int Vitesse { get; sl Organiser les instructions Using » 
} À, | Insérer un extrait. Ctri+K, X 
A. Entourer de... Ctri+K. S 


FIGURE 23.3 — Visual C# Express propose d’implémenter automatiquement l'interface 


Visual C# Express nous génère le code suivant : 


1| public class Voiture : IComparable 

2| € 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 
6 

7 

8 

9 


public int CompareTo(object obj) 
{ 


throw new NotImplementedException(); 
10 } 
ii | + 


Il s’agit de la signature de la méthode qui nous manque pour respecter le contrat 
de l'interface et d’un contenu que nous ne comprenons pas pour l'instant. Nous y 
reviendrons plus tard ; pour l’instant, vous n’avez qu’à supprimer la ligne : 


1 | throw new NotImplementedException(); 


Il ne reste plus qu’à écrire le code de la méthode. Pour ce faire, il faut définir un critère 
de tri. Trier des voitures n’a pas trop de sens, aussi nous dirons que nous souhaitons 
les trier suivant leurs vitesses. 


Pour respecter correctement le contrat, nous devons respecter la règle suivante qui se 

trouve dans la documentation de la méthode de l'interface : 

— si une voiture est inférieure à une autre, alors nous devons renvoyer une valeur 
inférieure à 0, disons -1 ; 

— si elle est égale, alors nous devons renvoyer 0; 

— si elle est supérieure, nous devons renvoyer une valeur supérieure à 0, disons 1. 


Ce qui donne : 


1| public int CompareTo(object obj) 

21 { 

3 Voiture voiture = (Voiture)obj; 

4 if (this.Vitesse < voiture.Vitesse) 
5 return -1; 

6 if (this.Vitesse > voiture.Vitesse) 
7 return 1; 

8 return O; 

91} 
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La comparaison s'effectue entre l’objet courant et un objet qui lui est passé en para- 
mètre. Pour que ce soit un peu plus clair, j’ai utilisé le mot-clé this qui permet de bien 
identifier l’objet courant et l’objet passé en paramètre. Comme il est facultatif, nous 
pouvons le supprimer. 


Vous aurez également remarqué que j'utilise un cast explicite avant de comparer. Ceci 
permet de renvoyer une erreur si jamais l’objet à comparer n’est pas du bon type. En 
effet, que devrais-je renvoyer si jamais l’objet qu’on me passe n’est pas une voiture ? 
Une erreur ! C’est très bien. 


Le code est suffisamment explicite pour que nous comprenions facilement ce que l’on 
doit faire : comparer les vitesses. 


Il est possible de simplifier grandement le code, car pour comparer nos deux voitures, 
nous effectuons la comparaison sur la valeur d’un entier, ce qui est plutôt trivial. 
D'autant plus que l’entier, en bon objet comparable, possède également la méthode 
CompareTo(). Ce qui fait qu’il est possible d'écrire notre méthode de comparaison de 
cette façon : 


1| public int CompareTo(object obj) 

2 |: € 

3 Voiture voiture = (Voiture)obj; 

4 return Vitesse.CompareTo(voiture.Vitesse); 
5| } 


En effet, Vitesse étant un type intégré, il implémente déjà correctement la comparai- 
son. C’est d’ailleurs pour ça que le tableau d’entier que nous avons vu précédemment 
a été capable de se trier facilement. 


En ayant implémenté cette interface, nous pouvons désormais trier des tableaux de 
Voiture : 


1] Voiture[] voitures = new Voiture[] { new Voiture { Vitesse = 
100 }, new Voiture { Vitesse = 40 }, new Voiture { Vitesse = 
10 }, new Voiture { Vitesse = 40 }, new Voiture { Vitesse = 
50 } }; 

2| Array.Sort (voitures); 

3| foreach (Voiture v in voitures) 

al { 

5 Console.WriteLine(v.Vitesse); 

6| } 


Ce qui affichera : 


10 


100 


Voilà pour le tri, mais si je peux me permettre, je trouve que ce code-là n’est pas très 
esthétique! J’y reviendrai un peu plus tard... 
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Nous avons donc implémenté notre première interface. Finalement, ce n’était pas si 
compliqué. Voyons à présent comment créer nos propres interfaces. Une interface se 
définit en C# comme une classe, sauf qu’on utilise le mot-clé interface à la place 
de class. En tant que débutant, vous aurez rarement besoin de créer des interfaces. 
Cependant, il est utile de savoir le faire. Par contre, il sera beaucoup plus fréquent que 
vos classes implémentent des interfaces existantes du framework .NET, comme nous 


venons de le faire. 


Voyons à présent comment créer une interface et examinons le code suivant : 


1| public interface IVolant 

21 { 

3 int NombrePropulseurs { get; set; } 
4 void Voler(); 

51 } 


Comme pour les classes, il est recommandé de créer les interfaces dans un 
fichier à part. Rappelez-vous également de la convention qui fait que les 
interfaces doivent commencer par un | majuscule. 


Nous définissons ici une interface IVolant qui possède une propriété de type int et 
une méthode Voler() qui ne renvoie rien. Voilà, c’est tout simple! 


Nous avons créé une interface. Les objets qui choisiront d’implémenter cette inter- 
face seront obligés d’avoir une propriété entière NombrePropulseurs et une méthode 
Voler() qui ne renvoie rien. Rappelez-vous que l’interface ne contient que le contrat 
et aucune implémentation. C'est-à-dire que nous ne verrons jamais de corps de mé- 
thode dans une interface ni de variables membres; uniquement des méthodes et des 
propriétés. Un contrat. 


Notez quand même qu’il ne faut pas définir de visibilité sur les membres d’une interface. 
Nous serons obligés de définir les visibilités en public sur les objets implémentant 
l'interface. 


Créons désormais deux objets Avion et Oiseau qui implémentent cette interface, ce 
qui donne : 


1| public class UÜiseau : IVolant 

2| € 

3 public int NombrePropulseurs { get; set; } 

4 

5 public void Voler() 

6 { 

7 Console.WriteLine("Je vole grâce à " + 
NombrePropulseurs + " ailes"); 

8 } 

o|} 

10 

11| public class Avion : IVolant 

12] À 

13 public int NombrePropulseurs { get; set; } 
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14 public void Voler() 

15 { 

16 Console.WriteLine ("Je vole grâce à " + 
NombrePropulseurs + " moteurs"); 

17 } 

18| } 


Grâce à ce contrat, nous savons maintenant que n'importe lequel de ces objets saura 
voler. 


Il est possible de traiter ces objets comme des objets volants, un peu comme ce que nous 
avions fait avec les classes mères, en utilisant l’interface comme type pour la variable. 
Par exemple : 


1| IVolant oiseau = new DUiseau { NombrePropulseurs = 2 }; 
2| oiseau.Voler(); 


Nous instancions vraiment un objet Oiseau, mais nous le manipulons en tant que 
IVolant. Un des intérêts dans ce cas sera de pouvoir manipuler des objets qui partagent 
un même comportement : 


Oiseau oiseau = new Oiseau { NombrePropulseurs = 2 }; 
Avion avion = new Avion { NombrePropulseurs = 4 }; 


List<IVolant> volants = new List<IVolant> { oiseau, avion }; 
foreach (IVolant volant in volants) 
{ 

volant.Voler(); 


D I OO Où À À ND 


} 
Ce qui produira : 


Je vole grâce 2 ailes 


Je vole grâce 4 moteurs 


Grâce à l’interface, nous avons pu mettre dans une même liste des objets différents, 
qui n’héritent pas entre eux mais qui partagent une même interface, c’est-à-dire un 
même comportement : IVolant. Pour accéder à ces objets, nous devrons utiliser leurs 
interfaces. 


Il sera possible quand même de caster nos IVolant en Avion ou en Oiseau, si jamais 
nous souhaitons rajouter une propriété propre à l’avion. Par exemple je rajoute une 
propriété NomDuCommandant à mon avion mais qui ne fait pas partie de l'interface : 


1| public class Avion : IVolant 

2| € 

3 public int NombrePropulseurs { get; set; } 

4 public string NomDuCommandant { get; set; } 

5 public void Voler() 

6 { 

7 Console.WriteLine("Je vole grâce à " + 
NombrePropulseurs + " moteurs"); 
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8 } 
o[ } 


Cela veut dire que l’objet Avion pourra affecter un nom de commandant mais qu’il ne 
sera pas possible d'y accéder par l'interface : 


1| IVolant avion = new Avion { NombrePropulseurs = 4, 
NomDuCommandant = "Nico" }; 

2| Console.WriteLine(avion.NomDuCommandant); // erreur de 
compilation 


L'erreur de compilation nous indique que IVolant ne possède pas de définition pour 
NomDuCommandant ; ce qui est vrai! 


Pour accéder au nom du commandant, nous pourrons tenter de caster nos IVolant en 
Avion. Si le cast est valide, alors nous pourrons accéder à notre propriété : 


1| Oiseau oiseau = new Oiseau { NombrePropulseurs = 2 }; 

2| Avion avion = new Avion { NombrePropulseurs = 4, 
NomDuCommandant = "Nico" }; 

3 

4! List<IVolant> volants = new List<IVolant> { oiseau, avion }; 

5| foreach (IVolant volant in volants) 

6| À 

“é volant.Voler(); 

8 Avion a = volant as Avion; 

9 if (a != null) 

10 { 

11 Console.WriteLine(a.NomDuCommandant); 

12 } 


13|1% 
Voilà, c’est tout simple et ça ressemble un peu à ce qu’on a déjà vu. 


Lorsque nous avons abordé la boucle foreach, j'ai dit qu'elle nous servait à 
parcourir des éléments « énumérables ». En disant ça, je disais en fait que 
la boucle foreach fonctionne avec tous les types qui implémentent l'inter- 
face IEnumerable. Maintenant que vous savez ce qu'est une interface, vous 
comprenez mieux ce à quoi je faisais vraiment référence. 


Il faut également noter que les interfaces peuvent hériter entre elles, comme c’est le cas 
avec les objets. C’est-à-dire que je vais pouvoir déclarer une interface IVolantMotorise 
qui hérite de l’interface IVolant. 


1| public interface IVolant 

2! { 

3 int NombrePropulseurs { get; set; } 

4 void Voler(); 

51 } 

6 

7| public interface IVolantMotorise : IVolant 
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void DemarrerLeMoteur(); 
10| } 


Ainsi, ma classe Avion qui implémentera IVolantMotorise devra obligatoirement im- 
plémenter les méthodes et propriétés de IVolant et la méthode de IVolantMotorise : 


1| public class Avion : IVolantMotorise 
2| € 

3 public void DemarrerLeMoteur () 

4 { 

5 } 

6 

7 public int NombrePropulseurs { get; set;} 
8 

9 public void Voler() 

10 { 

11 } 

12| } 


Enfin, et nous nous arrêterons là pour les interfaces, il est possible pour une classe 
d’implémenter plusieurs interfaces. Il suffira pour cela de séparer les interfaces par une 
virgule et d’implémenter bien sûr tout ce qu’il faut derrière. Par exemple : 


1| public interface IVolant 

2| { 

3 void Voler(); 

al} 

5 

6| public interface IRoulant 

7| { 

8 void Rouler); 

|} 

10 

11| public class Avion : IVolant, IRoulant 
12| À 

13 public void Voler() 

14 { 

15 Console.WriteLine ("Je vole"); 
16 } 

17 

18 public void Rouler() 

19 { 

20 Console.WriteLine ("Je Roule"); 
21 } 

22| } 
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Les classes et les méthodes abstraites 


Une classe abstraite est une classe particulière qui ne peut pas être instanciée. Concrè- 
tement, cela veut dire que nous ne pourrons pas utiliser l’opérateur new. 


De la même façon, une méthode abstraite est une méthode qui ne contient pas d’implé- 
mentation, c’est-à-dire pas de code. Pour être utilisables, les classes abstraites doivent 
être héritées et les méthodes redéfinies. 


En général, les classes abstraites sont utilisées comme classes de base pour d’autres 
classes. Ces classes fournissent des comportements mais n’ont pas vraiment d'utilité 
propre. Ainsi, les classes filles qui en héritent pourront bénéficier de leurs comporte- 
ments et devront éventuellement en remplacer d’autres. 


C’est comme une classe incomplète qui ne demande qu’à être complétée. 


Si une classe possède une méthode abstraite, alors la classe doit absolument être abs- 
traite. L’inverse n’est pas vrai, une classe abstraite peut posséder des méthodes non 
abstraites. Vous aurez rarement besoin d'utiliser les classes abstraites en tant que dé- 
butant mais elles pourront vous servir pour combiner la puissance des interfaces à 
l'héritage. 

Bon, voilà pour la théorie, passons un peu à la pratique! 


Rappelez-vous nos chiens et nos chats qui dérivent d’une classe mère Animal. Nous 
savons qu’un animal est capable de se déplacer. C’est un comportement qu’ont en com- 
mun tous les animaux. Il est tout à fait logique de définir une méthode SeDeplacer() 
au niveau de la classe Animal. Sauf qu’un chien ne se déplace pas forcément comme 
un dauphin. L'un a des pattes, l’autre des nageoires. Et même si le chien et le chat 
semblent avoir un déplacement relativement proche, ils ont chacun des subtilités. Le 
chat a un mouvement plus gracieux, plus félin. 


Bref, on se rend compte que nous sommes obligés de redéfinir la méthode SeDeplacer() 
dans chaque classe fille de la classe Animal. Et puis, de toute façon, sommes-nous 
vraiment capables de dire comment se déplace un Animal dans l’absolu ? 


La méthode SeDeplacer est une candidate parfaite pour une méthode abstraite. Rappelez- 
vous, la méthode abstraite ne possède pas d’implémentation et chaque classe fille de la 
classe possédant cette méthode abstraite devra la spécialiser. C’est exactement ce qu’il 
nous faut. 


Pour déclarer une méthode comme étant abstraite, il faut utiliser le mot-clé abstract 
et ne fournir aucune implémentation de la méthode, c’est-à-dire que la déclaration de 
la méthode doit se terminer par un point-virgule : 


1| public class Animal 

2| € 

3 public abstract void SeDeplacer(); 
al} 
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Si nous tentons de compiler cette classe, nous aurons l'erreur suivante : 


MaPremiereApplication.Animal.SeDeplacer()’ est abstrait, mais 


est contenu dans la classe non abstraite ? 
MaPremiereApplication.Animal ? 


Ah oui, c’est vrai, on a dit qu’une classe qui contient au moins une méthode abstraite 
était forcément abstraite. C’est le même principe que pour la méthode, il suffit d'utiliser 
le mot-clé abstract devant le mot-clé class : 


1| public abstract class Animal 

2| € 

3 public abstract void SeDeplacer(); 
al} 


Voilà, notre classe peut compiler tranquillement. 


Nous avons également dit qu’une classe abstraite pouvait contenir des méthodes concrètes 
et que c'était d’ailleurs une des grandes forces de ce genre de classes. En effet, il est 
tout à fait pertinent que des animaux puissent mourir. Et là, ça se passe pour tout le 
monde de la même façon : le cœur arrête de battre et on ne peut plus rien faire. C’est 
triste, mais c’est ainsi. 


Cela veut dire que nous pouvons créer une méthode Mourir () dans notre classe abs- 
traite qui possède une implémentation. 


Cela donne : 


1| public abstract class Animal 
2 |: € 

3 private Coeur coeur; 

4 public Animal () 

5 { 

6 coeur = new Coeur); 
7 } 

8 

9 public abstract void SeDeplacer(); 
10 

11 public void Mourir() 

12 { 

13 coeur.Stop(); 

14 } 

15| } 


Avec une classe Cœur qui ne fait pas grand-chose : 


1| public class Coeur 

21 { 

3 public void Battre() 

4 { 

5 Console.WriteLine ("Boom boom"); 
6 } 

Hé 
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8 public void Stop() 

9 { 

10 Console.WriteLine("Mon coeur s'arrête de battre"); 
11 } 

121} 


Comme prévu, il n’est pas possible d’instancier un objet Animal. Si nous tentons l’opé- 
ration : 


1 | Animal animal = new Animal); 


nous aurons l’erreur de compilation suivante : 


Impossible de créer une instance de la classe abstraite ou de 1? 


interface ?’MaPremiereApplication.Animal? 


Par contre, il est possible de créer une classe Chien qui dérive de la classe Animal : 


1| public class Chien : Animal 
2| € 
3| } 


Cette classe ne pourra pas compiler dans cet état car il faut obligatoirement redéfinir la 
méthode abstraite SeDeplacer(). Cela se fait en utilisant le mot-clé override, comme 
on l’a déjà vu. 

Vous aurez sûrement remarqué que la méthode abstraite n'utilise pas le mot-clé virtual 
comme cela doit absolument être le cas lors de la substitution d’une méthode dans une 
classe non-abstraite. Il est ici implicite et ne doit pas être utilisé, sinon nous aurons 
une erreur de compilation. Et puis cela nous arrange ; un seul mot-clé, c’est largement 
suffisant ! 


Nous devons donc spécialiser la méthode SeDeplacer, soit en écrivant la méthode à la 
main, soit en utilisant encore une fois notre ami Visual C# Express. 


Il suffit de faire un clic droit sur la classe dont hérite Chien (en l'occurrence Animal) 
et de cliquer sur Implémenter une classe abstraite (voir la figure 23.4). 


-ipublic class Chien : Animal 


{ Refactoriser » 
} Organiser les instructions Using » 
Implémenter une classe abstraite 
À, Insérer un extrait... Ctri+K, X 
À, Entourer de... Ctri+K, S 
53  Atteindre la définition F12 


LES SSSR ER rasown 


FIGURE 23.4 - Visual C# Express propose d’implémenter la classe abstraite 


Visual C# Express nous génère donc la signature de la méthode à substituer : 


1 | public class Chien : Animal 
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2| € 

3 public override void SeDeplacer() 

4 { 

5 throw new NotImplementedException(); 
6 } 

7|} 


Il ne reste plus qu’à écrire le corps de la méthode SeDeplacer, par exemple : 


1| public class Chien : Animal 

2| € 

3 public override void SeDeplacer() 

4 { 

5 Console.WriteLine("Waouf ! Je me déplace avec mes 4 
pattes"); 

6 } 

‘AR 


Ainsi, nous pourrons créer un objet Chien et le faire se déplacer puis le faire mourir 
car il hérite des comportements de la classe Animal. Paix à son âme. 


Chien max = new Chien); 
max.SeDeplacer(); 
max.Mourir(); 


Ce code affichera : 


Waouf ! Je me déplace avec mes 4 pattes 


Mon coeur s’arrête de battre 


Vous pouvez désormais vous rendre compte de la véracité de la phrase que j'ai écrite 
en introduction : 


« [Les classes abstraites] pourront vous servir pour combiner la puissance des interfaces 
à l'héritage. » 
En effet, la classe abstraite peut fournir des implémentations alors que l'interface ne 


propose qu’un contrat. Cependant, une classe concrète ne peut hériter que d’une seule 
classe mais peut implémenter plusieurs interfaces. 


La classe abstraite est un peu à mi-chemin entre l'héritage et l'interface. 


Les classes partielles 


Les classes partielles offrent la possibilité de définir une classe en plusieurs fois. En 
général, ceci est utilisé pour définir une classe sur plusieurs fichiers, si par exemple la 
classe devient très longue. Il pourra éventuellement être judicieux de découper la classe 
en plusieurs fichiers pour regrouper des fonctionnalités qui se ressemblent. 


On utilise pour cela le mot-clé partial. 


Par exemple, si nous avons un fichier qui contient la classe Voiture suivante : 
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public partial class Voiture 


1 
2| € 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 

6 


Nous pourrons compléter sa définition dans un autre fichier pour lui rajouter par 
exemple des méthodes : 


1| public partial class Voiture 

2| { 

3 public string Rouler () 

4 { 

5 return "Je roule à " + Vitesse + " km/h"; 
6 1, 

7|} 


À la compilation, Visual C4 Express réunit les deux classes en une seule et l’objet 
fonctionne comme toute autre classe qui ne serait pas forcément partielle. 


À noter qu'il faut impérativement que les deux classes possèdent le mot-clé partial 
pour que cela soit possible, sinon Visual C# Express générera une erreur de compila- 
tion : 


Modificateur partiel manquant sur la déclaration de type ? 


MaPremiereApplication.Voiture?” ; une autre déclaration 
partielle de ce type existe 


U z me dire qu n uper uti mm. nctionnalité. je vous dirai 
Vous alle e dire que ce n’est pas super utile co e fonctionnalité. Et je vous dira 
que vous avez raison. Sauf dans un cas particulier. 


Les classes partielles prennent de l'intérêt quand une partie du code de la classe est 
générée par Visual C# Express. C’est le cas pour la plupart des plateformes qui servent 
à développer de vraies applications. Par exemple ASP.NET pour un site internet, WPF 
pour une application Windows, Silverlight pour un client riche, etc. C’est aussi le cas 
lorsque nous générons de quoi permettre l’accès à une base de données. 


Dans ce cas-là, disons pour simplifier que Visual C# Express va nous générer tout 
un tas d'instructions pour nous connecter à la base de données ou pour récupérer 
des données. Toutes ces instructions seront mises dans une classe partielle que nous 
pourrons enrichir avec nos besoins. 


L'intérêt est que si nous générons une nouvelle version de notre classe, seul le fichier 
généré sera impacté. Si nous avions modifié le fichier pour enrichir la classe avec nos 
besoins, nous aurions perdu tout notre travail. Vu que, grâce aux classes partielles, ce 
code est situé dans un autre fichier, il n’est donc pas perdu. Pour notre plus grand 
bonheur ! 


À noter que le mot-clé partial peut se combiner sans problème avec d’autres mots-clés, 
comme abstract par exemple que nous venons de voir. 
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Il est fréquent aussi de voir des classes partielles utilisées quand plusieurs développeurs 
travaillent sur la même classe. Le fait de séparer la classe en deux fichiers permet de 
travailler sans se marcher dessus. 


Nous aurons l’occasion de voir des classes partielles générées plus tard dans l’ouvrage. 


Classes statiques et méthodes statiques 


Nous avons déjà vu le mot-clé static en première partie de ce livre. Il nous a bien 
encombrés ! Nous nous le sommes trimballé pendant un moment, puis il à disparu. 


Il est temps de revenir sur ce mot-clé afin de comprendre exactement de quoi il s’agit, 
maintenant que nous avons plus de notions et que nous connaissons les objets et les 
classes. 


Jusqu'à présent, nous avons utilisé le mot-clé static uniquement sur les méthodes et 
je l'ai expliqué vaguement en disant qu’il servait à indiquer que la méthode est toujours 
disponible et prête à être utilisée. 

Pas très convaincante mon explication... mais comme vous êtes polis, vous ne m'avez 
rien dit ! 


En fait, le mot-clé static permet d’indiquer que la méthode d’une classe n’appartient 
pas à une instance de la classe. Nous avons vu que jusqu’à présent, nous devions 
instancier nos classes avec le mot-clé new pour avoir des objets. 


Ici, static permet de ne pas instancier l’objet mais d’avoir accès à cette méthode en 
dehors de tout objet. Nous avons déjà utilisé beaucoup de méthodes statiques, je ne 
sais pas si vous avez fait attention, mais maintenant que vous connaissez les objets, la 
méthode suivante ne vous paraît pas bizarre ? 


1 | Console.WriteLine ("Bonjour"); 
Nous utilisons la méthode WriteLine de la classe Console sans avoir créé d’objet 
Console. Etrange. 


Il s’agit, vous l’aurez deviné, d’une méthode statique qui est accessible en dehors de 
toute instance de Console. D'ailleurs, la classe entière est une classe statique. Si nous 
essayons d’instancier la classe Console avec : 


1 | Console c = new Console(); 
Nous aurons les messages d’erreurs suivants : 


Impossible de déclarer une variable de type static System. 
Console? 


Impossible de créer une instance de la classe static ?’System. 
Console ? 


Nous avons dit que la méthode spéciale Main() est obligatoirement statique. Cela 
permet au CLR qui va exécuter notre application de ne pas avoir besoin d’instancier 
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la classe Program pour démarrer notre application. Il à juste à appeler la méthode 
Program.Main() afin de démarrer notre programme. 


Comme la méthode Main) est utilisable en dehors de toute instance de classe, elle ne 
peut appeler que des méthodes statiques. En effet, comment pourrait-elle appeler des 
méthodes d’un objet alors qu’elle n’en a même pas conscience. 


C’est pour cela que nous avons été obligés de préfixer chacune de nos premières mé- 
thodes par le mot-clé static. 


Revenons à nos objets. Ils peuvent contenir des méthodes statiques ou des variables 
statiques. Si une classe ne contient que des choses statiques alors elle peut devenir 
également statique. 


Une méthode statique est donc une méthode qui ne travaille pas avec les membres 
(variables ou autres) non statiques de sa propre classe. Rappelez-vous, un peu plus 
haut, nous avions créé une classe Math qui servait à faire des additions, afin d'illustrer 
le polymorphisme : 


1| public class Math 

2| { 

3 public int Addition(int a, int b) 
4 { 

5 return à + b; 

6 } 

7|} 


que nous utilisions de cette façon : 

1| Math math = new Math); 

2| int resultat = math.Addition(5, 6); 

Ici, la méthode addition sert à additionner deux entiers. Elle est complètement indé- 
pendante de la classe Math et donc des instances de l’objet Math. 


Nous pouvons donc en faire une méthode statique, pour cela il suffira de préfixer du 
mot-clé static son type de retour : 


1| public class Math 

2| { 

3 public static int Addition(int a, int b) 
4 { 

5 return à + b; 

6 } 

7|} 


Et nous pourrons alors utiliser l’addition sans créer d’instance de la classe Math, mais 
simplement en utilisant le nom de la classe suivi du nom de la méthode statique : 


1] int resultat = Math.Addition(5, 6); 


Exactement comme nous avons fait pour Console.WriteLine. 


De la même façon, nous pouvons rajouter d’autres méthodes, comme la multiplication : 
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1| public class Math 

2| { 

3 public static int Addition(int a, int b) 
4 { 

5 return à + b; 

6 } 

T 

8 public static long Multiplication(int a, int b) 
9 { 

10 return a * b; 

11 } 

12| } 


Nous l’appellerons de la même façon : 

1 | long resultat = Math.Multiplication(5, 6); 

À noter que la classe Math est toujours instanciable mais qu’il n’est pas possible d’ap- 
peler les méthodes qui sont statiques depuis un objet Math. Le code suivant : 

1| Math math = new Math); 


2| long resultat = math.Multiplication(5, 6); 


provoquera l'erreur de compilation : 


Le membre ’MaPremiereApplication.Math.Multiplication(int, int)? 


est inaccessible avec une référence d’instance ; qualifiez-le 
avec un nom de type 


Ok, mais à quoi ça sert de pouvoir instancier un objet Math si nous ne pouvons 
accéder à aucune de ses méthodes, vu qu'elles sont statiques ? 


Absolument à rien! Si une classe ne possède que des membres statiques, alors il est 
possible de rendre cette classe statique grâce au même mot-clé. 


Celle-ci deviendra non-instanciable, comme la classe Console : 


1| public static class Math 

2| { 

3 public static int Addition(int a, int b) 
4 { 

5 return à + b; 

6 } 

7|} 


Ainsi, si nous tentons d’instancier l’objet Math, nous aurons une erreur de compilation. 


En général, les classes statiques servent à regrouper des méthodes utilitaires qui par- 
tagent une même fonctionnalité. Ici, la classe Math permettrait de ranger toutes les 
méthodes du style addition, multiplication, racine carrée, etc. 
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Ah, on me fait signe que cette classe existe déjà dans le framework .NET et qu’elle 
s'appelle également Math. Elle est rangée dans l’espace de nom System. Souvenez-vous, 
nous l’avons utilisée pour calculer la racine carrée. Cette classe est statique, c’est la 
version aboutie de la classe que nous avons commencé à écrire. 


Par contre, ce n’est pas parce qu’une classe possède des méthodes statiques qu’elle est 
obligatoirement statique. Il est aussi possible d’avoir des membres statiques dans une 
classe qui possède des membres non statiques. 


Par exemple notre classe Chien, qui possède un prénom et qui sait aboyer : 


1| public class Chien 

2| € 

3 private string prenom; 

4 

5 public Chien(string prenomDuChien) 
6 { 

7 prenom = prenomDuChien; 

8 } 

9 

10 public void Aboyer() 

11 { 

12 Console.WriteLine("Wouaf ! Je suis " + prenom); 
13 } 

141} 


pourrait posséder une méthode permettant de calculer l’âge d’un chien dans le référen- 
tiel des humains. 


Comme beaucoup le savent, il suffit de multiplier l’âge du chien par 7! 


1| public class Chien 

2| € 

3 private string prenom; 

4 

5 public Chien(string prenomDuChien) 

6 { 

7 prenom = prenomDuChien; 

8 } 

9 

10 public void Aboyer() 
11 { 
12 Console.WriteLine("Wouaf ! Je suis " + prenom); 
13 } 
14 
15 public static int CalculerAge(int ageDuChien) 
16 É 
17 return ageDuChien * 7; 
18 } 
19! } 


Ici, la méthode CalculerAge() est statique car elle ne travaille pas directement avec 
une instance d’un chien. 
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Nous pouvons l’appeler ainsi : 


Chien hina = new Chien("Hina"); 

hina.Aboyer(); 

int ageReferentielHomme = Chien.CalculerAge (4); 
Console .WriteLine (ageReferentielHomme) ; 


& © ND OH 


Vous aurez ainsi dans la console : 


Waouf ! Je suis Hina 


28 


Vous me direz qu’il est possible de faire en sorte que la méthode travaille sur une 
instance d’un objet Chien, ce qui serait peut-être plus judicieux ici. Il suffirait de 
rajouter une propriété Age au Chien et de transformer la méthode pour qu’elle ne soit 
plus statique. Ce qui donnerait : 


1| public class Chien 

2|{ 

3 private string prenom; 

4 

5 public int Age { get; set; } 
6 

7 public Chien(string prenomDuChien) 
8 { 

9 prenom = prenomDuChien; 
10 } 

11 

12 public void Aboyer() 

13 { 

14 Console.WriteLine("Wouaf ! Je suis " + prenom); 
15 } 

16 

17 public int CalculerAge() 

18 { 

19 return Age * 7; 

20 } 

21| } 


Que l’on pourrait appeler de cette façon : 


1| Chien hina = new Chien('"Hina") { Age = 5 }; 
2| int ageReferentielHomme = hina.CalculerAge(); 
3| Console.WriteLine (ageReferentielHomme) ; 


Ici, c’est plus une histoire de conception. C’est à vous de décider, mais sachez que 
c’est possible. Il est également possible d'utiliser le mot-clé static avec des propriétés. 
Imaginons que nous souhaitions avoir un compteur sur le nombre d’instances de la classe 
Chien. Nous pourrions créer une propriété statique de type entier qui s’incrémente à 
chaque fois que l’on crée un nouveau chien : 
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1| public class Chien 

2| € 

3 public static int NombreDeChiens { get; set; } 
4 

5 private string prenom; 

6 

7 public Chien(string prenomDuChien) 

8 { 

9 prenom = prenomDuChien; 

10 NombreDeChiens++; 

11 } 

12 

13 public void Aboyer() 

14 { 

15 Console.WriteLine("Wouaf ! Je suis " + prenom); 
16 } 

17 | } 


Ici, la propriété NombreDeChiens est statique et s’incrémente à chaque passage dans le 
constructeur. 


Ainsi, si nous créons plusieurs chiens : 


Chien chieni = new Chien("Max"); 
Chien chien2 = new Chien("Hina"); 
Chien chien3 = new Chien("Laika"); 


où & © ND 


Console.WriteLine(Chien.NombreDeChiens); 


Nous pourrons voir combien de fois nous sommes passés dans le constructeur : 


À noter que pour une variable statique, cela se passe de la même façon qu'avec les 
propriétés statiques. 


Les classes internes 


Les classes internes | sont un mécanisme qui permet d’avoir des classes définies à l’in- 


térieur d’autres classes. 


Cela peut être utile si vous souhaitez restreindre l’accès d’une classe uniquement à sa 
classe mère. 


Par exemple : 


1| public class Chien 
2| € 


3 private Coeur coeur = new Coeur(); 


1. Classe interne se dit nested class en anglais. 
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4 

5 public void Mourir() 

6 { 

7 coeur.Stop(); 

8 is 

9 

10 private class Coeur 

11 € 

12 public void Stop() 
13 { 

14 Console.WriteLine("The end"); 
15 F 

16 } 

17| } 


Ici, la classe Coeur ne peut être utilisée que par la classe Chien car elle est privée. 
Nous pourrions également mettre la classe en protected afin qu’une classe dérivée de 
la classe Chien puisse également utiliser la classe Coeur : 


1| public class ChienSamois : Chien 

2| € 

3 private Coeur autreCoeur = new Coeur (); 
al} 


Avec ces niveaux de visibilité, une autre classe comme la classe Chat ne pourra pas se 
servir de ce cœur. Si nous mettons la classe Coeur en public ou internal, elle sera 
utilisable par tout le monde, comme une classe normale. Dans ce cas, pour nos chats, 
nous pourrions avoir : 


1| public class Chat 

2| € 

3 private Chien.Coeur coeur = new Chien.Coeur(); 
4 

5 public void Mourir() 

6 { 

7 coeur.Stop(); 

8 } 

o[} 


Notez que nous préfixons la classe par le nom de la classe qui contient la classe Coeur. 
Dans ce cas, l'intérêt d'utiliser une classe interne est moindre. Cela permet éventuelle- 
ment de regrouper les classes de manière sémantique, et encore, c’est plutôt le but des 
espaces de nom. Voilà pour les classes internes. C’est une fonctionnalité qui est peu 
utilisée, mais vous la connaissez désormais ! 


Les types anonymes et le mot-clé var 


Le mot-clé var est un truc de feignant ! 
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Il permet de demander au compilateur de déduire le type d’une variable au moment 
où nous la déclarons. Par exemple le code suivant : 


1 
2 


var prenom = "Nicolas"; 
var age = 30; 


est équivalent au code suivant : 


1 
2 


string prenom = "Nicolas"; 
int age = 30; 


Le mot-clé var sert à indiquer que nous ne voulons pas nous préoccuper de ce qu'est 
le type et que c’est au compilateur de le trouver. 


Cela implique qu’il faut que la variable soit initialisée (et non nulle) au moment où elle 
est déclarée afin que le compilateur puisse déduire le type de la variable, en l'occurrence, 
il devine qu’en lui affectant « Nicolas », il s’agit d’une chaîne de caractères. 


Je ne recommande pas l’utilisation de ce mot-clé car le fait de ne pas mettre le type de 
la variable fait perdre de la clarté au code. Ici, c’est facile. On déduit facilement nous 
aussi que le type de prenom est string et que le type de age est int. 


Mais c’est aussi parce qu’on est trop fort ! 
Par contre, si jamais la variable est initialisée grâce à une méthode : 


1 | var calcul = GetCalcul(); 


il va falloir aller regarder la définition de la méthode afin de savoir le type de retour et 
connaître ainsi le type de la variable. Des contraintes dont on n’a pas besoin alors qu’il 
est aussi simple d'indiquer le vrai type et que c’est d’autant plus lisible. 


© Mais alors, pourquoi en parler ? 


Parce que ce mot-clé peut servir dans un cas précis, celui des types anonymes. Lorsqu'il 
est conjointement utilisé avec l'opérateur new, il permet de créer des types anonymes, 
c’est-à-dire des types dont on n’a pas défini la classe au préalable. Une classe sans nom. 


Cette classe est déduite grâce à son initialisation : 
1 | var unePersonneAnonyme = new { Prenom = "Nico", Age = 30 }; 
Ici nous créons une variable qui contient une propriété Prenom et une propriété Age. 


Forcément, il est impossible de donner un type à cette variable vu qu’elle n’a pas de 
définition. C’est pour cela qu’on utilise le mot-clé var. 


Ici, on sait juste que la variable unePersonneAnonyme possède deux propriétés, un 
prénom et un âge. En interne, le compilateur va générer un nom de classe pour ce type 
anonyme, mais il n’a pas de sens pour nous. En l’occurrence, si nous écrivons le code 
suivant où GetType() (hérité de la classe object) renvoie le nom du type : 


1| var unePersonneAnonyme = new { Prenom = "Nico", Age = 30 }; 
2| Console.WriteLine (unePersonneAnonyme.GetType()); 
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nous aurons dans la console le type de notre variable : 


<>f__AnonymousTypeO ‘2[System.String,System.Int32] 


Ce qui est effectivement sans intérêt pour nous! 


Jusqu'ici, les types anonymes peuvent sembler ne pas apporter d'intérêt, tant il est 
simple de définir une classe Personne possédant une propriété Prenom et une propriété 
Age. 


Mais cela permet d'utiliser ces classes comme des classes à usage unique lorsque nous 
ne souhaitons pas nous encombrer d’un fichier possédant une classe qui va nous servir 
uniquement à un seul endroit. 


Paresse, souci de clarté du code... tout ceci est un peu mêlé dans la création d’un 
type anonyme. À vous de l’utiliser quand bon vous semble. Vous verrez plus tard que 
les types anonymes sont souvent utilisés dans les méthodes d’extensions Linq. Nous en 
reparlerons ! 


À noter que lorsque nous créons un tableau, par exemple : 


1| string[] jours = new string[] { "Lundi", "Mardi", "Mercredi", 


Jeudi", "Vendredi", "Samedi", "Dimanche" }; 


il est également possible de l’écrire ainsi : 


1| string[] jours = new[] { "Lundi", "Mardi", "Mercredi", "Jeudi", 


"Vendredi", "Samedi", "Dimanche" }; 


c’est-à-dire sans préciser le type du tableau après le new. Le compilateur déduit le type 
à partir de l’initialisation. Ce n’est pas un type anonyme en soi, mais le principe de 
déduction est le même. 


En résumé 


— Une interface est un contrat que s'engage à respecter un objet. 

— Il est possible d’implémenter plusieurs interfaces dans une classe. 

— Une classe abstraite est une classe qui possède au moins une méthode ou propriété 
abstraite. Elle ne peut pas être instanciée. 

— Une classe concrète dérivant d’une classe abstraite est obligée de substituer les 
membres abstraits. 

— Il est possible de créer des types anonymes grâce à l’opérateur new suivi de la des- 
cription des propriétés de ce type. Les instances sont manipulées grâce au mot-clé 
var. 
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ae 24 


TP : programmation orientée objet 


Difficulté : 3 


ans ce TP, nous allons essayer de mettre en pratique ce que nous avons appris en 
programmation orientée objet avec le C#. Il est difficile d'avoir un exercice faisant 
appel à toutes les notions que nous avons apprises. Aussi, certaines sont laissées de 
côté. 
Le TP est intéressant pour s'entraîner à créer des classes, à manipuler l'héritage et à se 
confronter à des situations un peu différentes de la théorie. 


Le but de ce TP est de créer une mini application de gestion bancaire où nous pourrons 
gérer des comptes pouvant effectuer des opérations bancaires entre eux. Ne vous attendez 
pas non plus à refaire les applications de la Banque de France; on est là pour s'entraîner! 


C# 
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Instructions pour réaliser le TP 


Voici l'énoncé de la première partie du TP : dans notre mini application bancaire 
tous les montants utilisés seront des décimaux. Nous allons devoir gérer des comptes 
bancaires. Un compte est composé : 


— d’un Solde (calculé, mais non modifiable) ; 

— d’un Propriétaire (nom du propriétaire du compte) ; 

— d’une liste d'opérations internes permettant de garder l’historique du compte, non 
accessible par les autres objets ; 

— d’une méthode permettant de Crediter() le compte, prenant une somme en para- 
mètre ; 

— d’une méthode permettant de Crediter() le compte, prenant une somme et un 
compte en paramètres, créditant le compte et débitant le compte passé en para- 
mètres ; 

— d’une méthode permettant de Debiter() le compte, prenant une somme en para- 
mètre ; 

— d’une méthode permettant de Debiter() le compte, prenant une somme et un 
compte bancaire en paramètres, débitant le compte et créditant le compte passé 
en paramètre ; 

— d’une méthode qui permet d’afficher le résumé d’un compte. 


Un compte courant est une sorte de compte et se compose : 


— de tout ce qui compose un compte ; 
— d’un découvert autorisé, non modifiable, défini à l’ouverture du compte. 


Le résumé d’un compte courant affiche le solde, le propriétaire, le découvert autorisé 
ainsi que les opérations sur le compte. 


Un compte épargne entreprise est une sorte de compte et se compose : 


— de tout ce qui compose un compte ; 
— d’un taux d’abondement, défini à l'ouverture du compte en fonction de l’ancien- 
neté du salarié. Un taux est un double compris entre 0 et 1 (5% — 0.05). 


Le solde doit tenir compte du taux d’abondement, mais l’abondement n’est pas calculé 
lors des transactions car l’entreprise verse l’abondement quand elle le souhaite. 


Le résumé d’un compte épargne entreprise affiche le solde, le propriétaire, le taux 
d’abondement ainsi que les opérations sur le compte. 


Une opération bancaire est composée : 


— d’un montant ; 
— d’un type de mouvement, crédit ou débit. 


Voilà! Vous allez maintenant devoir créer une telle application avec les informations 
suivantes : 


— le compte courant de Nicolas a un découvert autorisé de 2000 € ; 
— le compte épargne entreprise de Nicolas à un taux de 2% ; 
— le compte courant de Jérémie a un découvert autorisé de 500 € ; 
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— Nicolas touche son salaire de 100 € (pas cher payé)! 

il fait le plein de sa voiture : 50 €; 

— il met de côté sur son compte épargne entreprise la coquette somme de 20 €; 

— il reçoit un cadeau de la banque de 100 €, car il a ouvert son compte épargne pendant 
la période promotionnelle ; 

— il remet ses 20 € sur son compte bancaire, car finalement, il ne se sent pas trop en 
sécurité ! 

— Jérémie achète un nouveau PC : 500 €; 

— Jérémie rembourse ses dettes à Nicolas : 200 €. 


L'application doit indiquer les soldes de chacun de ces comptes : 


Solde compte courant de Nicolas : 250 


Solde compte épargne de Nicolas : 102,00 
Solde compte courant de Jérémie : -700 


Ensuite, nous afficherons le résumé du compte courant de Nicolas et de son compte 
épargne entreprise. Ce qui nous donnera : 


Résumé du compte de Nicolas 
ke 2 he He ke DK 0 2e 2H He DK DK DK 2 He Ke DK DK 0e 2e ke ke DK DK ee ke ke DK DK ok 2 ke ke 
Compte courant de Nicolas 

Solde : 250 

Découvert autorisé : 2000 


Opérations 
+100 
-50 
-20 
+20 
+200 
HO OKOHOKEHEOKKOOHOKEOOKOROKOKEOOOKOOKEOOKEOEO K 


Résumé du compte épargne de Nicolas 
HHHHHHHHHHHHHHHAHAHHHHHHAHHHHHHAHHHHH 
Compte épargne entreprise de Nicolas 
Solde : 102,00 
Taux  : 0,02 


Opérations 
+20 
+100 
-20 
HEHHHEHEHEHEHE AH AH AH HAE HE AH HE HA AH AH OH AAA 


Bon, j'ai bien expliqué, ce n’est pas si compliqué, sauf si bien sûr vous n’avez pas lu 
ou pas compris les chapitres précédents. Dans ce cas, n’hésitez pas à y rejeter un coup 
d'œil. 


Sinon, il suffit de bien décomposer tout en créant les classes les unes après les autres. 
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Bon courage...! 


Correction 


Ne regardez pas tout de suite la correction ! Prenez votre temps pour faire ce TP, car il 
est important de savoir bien manipuler les classes. Vous allez faire ça très régulièrement ; 
ça doit devenir un réflexe, il faut donc pratiquer ! 


Toujours est-il que voici ma correction. Le plus dur était certainement de modéliser 
correctement l’application. 


Il y à plusieurs solutions bien sûr pour créer ce petit programme, voici celle que je 
propose. 


Tout d’abord, nous devons manipuler des comptes courants et des comptes épargne 
entreprise, qui sont des sortes de comptes. 


On en déduit qu’un compte courant hérite d’un compte. De même, un compte épargne 
entreprise hérite également d’un compte. D'ailleurs, un compte a-t-il vraiment une 
raison d’être à part entière ? Nous ne créons jamais de compte « généraliste », seulement 
des comptes spécialisés. Nous allons donc créer la classe Compte en tant que classe 
abstraite. 


Ce qui nous donnera les classes suivantes : 


public abstract class Compte 
{ 
} 


public class CompteCourant : Compte 
{ 
} 
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public class CompteEpargneEntreprise : Compte 
{ 
} 


rh 
h © 


Nous avons également dit qu’un compte était composé d’une liste d'opérations qui 
possèdent un montant et un type de mouvement. Le type de mouvement pouvant 
prendre deux valeurs, il paraît logique d’utiliser une énumération : 


1| public enum Mouvement 
2| € 

3 Credit, 

4 Debit 

5| } 


La classe d'opération étant : 


1| public class Operation 
2| € 
3 public Mouvement TypeDeMouvement { get; set; } 


CORRECTION 


4 public decimal Montant { get; set; } 
} 


Voilà pour la base. 


Ensuite, la classe Compte doit posséder un nom de propriétaire et un solde qui peut 
être redéfini dans la classe CompteEpargneEntreprise. Ce solde est calculé à partir 
d’une liste d'opérations. Le solde est donc une propriété en lecture seule, virtuelle car 


nécessitant d’être redéfinie : 


1| public abstract class Compte 

2| € 

3 protected List <Operation> listeUperations; 

4 public string Proprietaire { get; set; } 

5 

6 public virtual decimal Solde 

7 { 

8 get 

9 Le 

10 decimal total = O0; 

11 foreach (Operation operation in listeUperations) 

12 { 

13 if (operation.TypeDeMouvement == Mouvement. 
Credit) 

14 total += operation.Montant; 

15 else 

16 total -= operation.Montant; 

17 } 

18 return total; 

19 LL 

20 } 

21| } 


La liste des opérations est une variable membre, déclarée en protected car nous allons 
en avoir besoin dans la classe CompteCourant qui en hérite, afin d’afficher un résumé 
des opérations. 


La propriété Solde n’est pas très compliquée en soit, il suffit de parcourir la liste des 
opérations et en fonction du type de mouvement, ajouter ou retrancher le montant. 
Comme c’est une propriété en lecture seule, seul le get est défini. 


N'oubliez pas que la liste doit être initialisée avant d’être utilisée, sinon nous aurons 
une erreur. Le constructeur est un endroit approprié pour le faire : 


1| public abstract class Compte 

2| € 

3 11 L: 2.1 

4 // [Code précédent enlevé pour plus de lisibilité] 
5 FE Les 

6 

7 public Compte() 

8 { 
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9 listeOperations = new List<Operation>(); 


La classe doit ensuite posséder des méthodes permettant de créditer ou de débiter le 
compte, ce qui donne : 


1| public abstract class Compte 

2| € 

3 // T...] 

4 // [Code précédent enlevé pour plus de lisibilité] 

5 ESS | 

6 

7 public void Crediter(decimal montant) 

8 { 

9 Operation operation = new Üperation { Montant = montant 
, TypeDeMouvement = Mouvement .Credit}; 

10 listeU0perations.Add(operation); 

11 } 

12 

13 public void Crediter(decimal montant, Compte compte) 

14 { 

15 Crediter (montant); 

16 compte.Debiter (montant); 

17 } 

18 

19 public void Debiter(decimal montant) 

20 { 

21 Operation operation = new Operation { Montant = montant 
, TypeDeMouvement = Mouvement .Debit }; 

22 liste0perations.Add(operation); 

23 } 

24 

25 public void Debiter(decimal montant, Compte compte) 

26 { 

27 Debiter (montant); 

28 compte.Crediter (montant); 

29 } 

30 | } 


Le principe est de créer une opération et de l’ajouter à la liste des opérations. L’astuce 
ici consiste à réutiliser les méthodes de la classe pour écrire les autres formes des 
méthodes, ce qui simplifie le travail et facilitera les éventuelles futures opérations de 


maintenance. 


Enfin, chaque classe dérivée de la classe Compte doit afficher un résumé du compte. 
Nous pouvons donc forcer ces classes à devoir implémenter cette méthode en utilisant 
une méthode abstraite : 


1| public abstract class Compte 
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14. 1: 41] 
// [Code précédent enlevé pour plus de lisibilité] 
FE sel 


public abstract void AfficherResume(); 


Œ I OO Où À 


} 


Voilà pour la classe Compte. En toute logique, c’est elle qui contient le plus de méthodes 
afin de factoriser le maximum de code commun dans la classe mère. 


Passons à la classe CompteEpargneEntreprise, qui hérite de la classe Compte. Elle 
possède un taux d’abondement qui est défini à la création du compte. Il est donc 
ici intéressant de forcer le positionnement de ce taux lors de la création de la classe, 
c’est-à-dire en utilisant un constructeur avec un paramètre : 


1] public class CompteEpargneEntreprise : Compte 

2| € 

3 private double tauxAbondement ; 

4 

5 public CompteEpargneEntreprise(double taux) 
6 À 

7 tauxAbondement = taux; 

8 } 

9} 


Ce taux est utilisé pour calculer le solde du compte en faisant la somme de toutes les 
opérations et en appliquant le taux: ce qui revient à appeler le calcul du solde de la 
classe mère et à lui appliquer le taux. Nous substituons donc la propriété Solde et 
utilisons le calcul fait dans la classe Compte : 


1] public class CompteEpargneEntreprise : Compte 

2| € 

3 71 +: 

4 // [Code précédent enlevé pour plus de lisibilité] 
5 {1 ze] 
6 

7 

8 

9 


public override decimal Solde 


{ 
get 
10 { 
11 return base.Solde * (decimal)(1 + tauxAbondement) ; 
12 } 
13 } 
141} 


Rien de plus simple, en utilisant le mot-clé base pour appeler le solde de la classe mère. 
Vous noterez également que nous avons eu besoin de caster le taux qui est un double 
afin de pouvoir le multiplier à un décimal. 


Enfin, cette classe se doit de fournir une implémentation de la méthode AfficherResume () : 


1 | public class CompteEpargneEntreprise : Compte 
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2| € 

3 CP Lis 

4 // [Code précédent enlevé pour plus de lisibilité] 

5 // T...1 

6 

7 public override void AfficherResume() 

8 { 

9 Console.WriteLine (" 
HAE HE HA HARMAN) 

10 Console.WriteLine ("Compte épargne entreprise de " + 
Proprietaire); 

11 Console.WriteLine ("\tSolde : " + Solde); 

12 Console.WriteLine("\tTaux : " + tauxAbondement) ; 

13 Console.WriteLine("\n\n0Opérations :"); 

14 foreach (Operation operation in listeUperations) 

15 { 

16 if (operation.TypeDeMouvement == Mouvement.Credit) 

17 Console.Write("\t+"); 

18 else 

19 Console.Write("\t-"); 

20 Console.WriteLine (operation.Montant); 

21 } 

22 Console.WriteLine (" 
HAMMAM HAMMAM HHHHHHHAHHHHHHHHHN) : 

23 } 

24| } 


Il s’agit d’une banale méthode d'affichage où nous parcourons la liste contenant les 
opérations et affichons le montant en fonction du type de mouvement. 


Voilà pour la classe CompteEpargneEntreprise. 


Il ne reste plus que la classe CompteCourant qui doit également dériver de Compte : 


1| public class CompteCourant : Compte 
2| € 
3| } 


Un compte courant doit posséder un découvert autorisé, défini à la création du compte 
courant. Nous utilisons comme avant un constructeur avec un paramètre : 


1| public class CompteCourant : Compte 

2| € 

3 private decimal decouvertAutorise; 

4 

5 public CompteCourant(decimal decouvert) 
6 { 

7 decouvertAutorise = decouvert; 

8 } 

a| } 


Il ne restera plus qu’à fournir une implémentation de la méthode d'affichage du résumé : 
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10 
11 
12 


13 
14 
15 
16 
17 
18 
19 
20 
21 
22 


23 
24 


public class CompteCourant : Compte 
{ 
JE za: 
// [Code précédent enlevé pour plus de lisibilité] 
PNR SI 
public override void AfficherResume() 
{ 
Console.WriteLine(" 

KR HOHOK KO OH OH OO KE OK EEE EE EE M) 
Console.WriteLine ("Compte courant de " + Proprietaire); 
Console.WriteLine("\tSolde : " + Solde); 
Console.WriteLine("\tDécouvert autorisé : " + 

decouvertAutorise); 
Console.WriteLine("\n\nOpérations :"); 
foreach (Operation operation in listeU0perations) 
{ 

if (operation.TypeDeMouvement == Mouvement .Credit) 

Console.Write("\t+"); 
else 
Console.Write("\t-"); 
Console.WriteLine (operation.Montant); 
} 
Console.WriteLine(" 
KA HO OO OO ID OO OO EEE) 5 
} 


Voilà pour nos objets. Cela en fait un petit paquet. Mais ce n’est pas fini, il faut 
maintenant créer des comptes et faire des opérations. 


L'’énoncé consiste à faire les instanciations suivantes, depuis notre méthode Main() : 


1 


co 


CompteCourant compteNicolas = new CompteCourant (2000) { 


Proprietaire = "Nicolas" }; 
CompteEpargneEntreprise compteEpargneNicolas = new 
CompteEpargneEntreprise(0.02) { Proprietaire = "Nicolas" }; 
CompteCourant compteJeremie = new CompteCourant (500) { 
Proprietaire = "Jérémie" }; 


compteNicolas.Crediter (100); 
compteNicolas.Debiter (50); 


compteEpargneNicolas.Crediter (20, compteNicolas); 
compteEpargneNicolas.Crediter (100); 


compteEpargneNicolas.Debiter(20, compteNicolas); 


compteJeremie.Debiter (500); 
compteJeremie.Debiter (200, compteNicolas); 
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16| Console.WriteLine ("Solde compte courant de " + compteNicolas. 
Proprietaire + " : " + compteNicolas.Solde); 

17] Console.WriteLine ("Solde compte épargne de " + 
compteEpargneNicolas.Proprietaire + " : " + 
compteEpargneNicolas.Solde); 

18| Console.WriteLine ("Solde compte courant de " + compteJeremie. 
Proprietaire + " : " + compteJeremie.Solde); 


19| Console.WriteLine("\n"); 


Rien d’extraordinaire. 
Il ne reste plus qu’à afficher le résumé des deux comptes demandés : 


Console.WriteLine ("Résumé du compte de Nicolas"); 
compteNicolas.AfficherResume() ; 
Console.WriteLine("\n"); 


Console.WriteLine ("Résumé du compte épargne de Nicolas"); 
compteEpargneNicolas.AfficherResume () ; 
Console.WriteLine("\n"); 


NN OO Où À © NN H 


Nous en avons fini avec la première partie du TP. 


Si vous avez bien compris comment construire des classes, comment les faire hériter 
entre elles et les interfaces, ce TP n’a pas dû vous poser trop de problèmes. Il existe bien 
sûr différentes solutions pour résoudre ce problème, de façon plus ou moins complexe. 


Aller plus loin 


Il y a plusieurs choses qui me dérangent dans ce code. La première est la répétition. 
Dans les deux méthodes AfficherResume() des classes respectives : CompteCourant 
et CompteEpargneEntreprise, on affiche les opérations de la même façon. Si jamais 
je dois modifier ce code (bug ou évolution), je devrais le faire dans les deux classes, 
au risque d’en oublier une. Il faut donc factoriser ce code. Il existe plusieurs solutions 
simples de factoriser ce code. La première est d'utiliser une méthode statique pour 
afficher la liste des opérations. Cette méthode pourrait être située dans une classe 
utilitaire qui prend en paramètre la liste des opérations : 


1| public static class Helper 


2|{ 

3 public static void AfficheUperations (List <Operation> 
operations) 

4 { 

5 Console.WriteLine("\n\n0Opérations :"); 

6 foreach (Operation operation in operations) 

7 { 

8 if (operation.TypeDeMouvement == Mouvement.Credit) 

9 Console.Write("\t+"); 

10 else 

11 Console.Write("\t-"); 
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12 Console.WriteLine (operation.Montant); 
13 } 

14 } 

15| } 


Il ne reste plus qu’à utiliser cette méthode statique depuis nos méthodes AfficherResume(), 
par exemple pour le compte épargne entreprise : 


1| public override void AfficherResume() 

2| € 

3 Console.WriteLine ("HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHN) ; 

4 Console.WriteLine ("Compte épargne entreprise de " + 
Proprietaire); 

Console.WriteLine("\tSolde : " + Solde); 

Console.WriteLine("\tTaux : " + tauxAbondement) ; 

Helper.AfficheUperations(listeUperations); 

Console.WriteLine ("HHHHHHHHHHHHHHHHHHHHHAHHHHHHHHHHHHHHHHN) ; 


© © 1 OO 


} 


La deuxième solution est de créer la méthode AfficheUperations() dans la classe 
abstraite Compte, ce qui permet de s'affranchir de passer la liste en paramètre vu que 
la classe Compte la connaît déjà : 


1| protected void AfficheUperations() 

2| € 

3 Console.WriteLine("\n\n0Opérations :"); 

4 foreach (Operation operation in listeU0perations) 
5 { 

6 if (operation.TypeDeMouvement == Mouvement.Credit) 
7 Console.Write("\t+"); 

8 else 

9 Console.Write("\t-"); 

10 Console.WriteLine(operation.Montant); 

11 x 

12| } 


La méthode a tout intérêt à être déclarée en protected, afin qu’elle puisse servir aux 
classes filles mais pas à l’extérieur. Elle s’utilisera de cette façon, par exemple dans la 
classe CompteEpargneEntreprise : 


1| public override void AfficherResume() 

2| { 

3 Console.WriteLine ("HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHN) ; 

4 Console.WriteLine ("Compte épargne entreprise de " + 
Proprietaire); 

Console.WriteLine("\tSolde : " + Solde); 

Console.WriteLine("\tTaux : " + tauxAbondement) ; 

AfficheOperations(); 

Console.WriteLine ("HHHHHHHHHHHHHHHHHHHHHAHHHHHHHHHHHHHHHHN) ; 


© © 1 OO 
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Dès que l’on peut factoriser du code, il ne faut pas hésiter. Si nous avons demain 
besoin de créer un nouveau type de compte, nous serons ravis de pouvoir nous servir 
de méthodes toutes prêtes nous simplifiant le travail. 


Il pourrait être intéressant également d’encapsuler l’enregistrement d’une opération 
dans une méthode. Ça permettrait de moins se répéter (même si ici, le code est vraiment 
petit) mais surtout de séparer la logique d'enregistrement d’une opération afin que cela 
soit plus facile ultérieurement à modifier, maintenir ou complexifier. Par exemple, via 
une méthode : 


1| private void Enregistrer0Operation(Mouvement typeDeMouvement, 
decimal montant) 


2| € 

3 Operation operation = new Operation { Montant = montant, 
TypeDeMouvement = typeDeMouvement}; 

4 liste0perations.Add(operation); 

5| } 


Cela permet également de faire en sorte que le type de mouvement soit un paramètre 
de la méthode. 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code ) 


Code web : 729727 


Deuxième partie du TP 


Nous voici maintenant dans la deuxième partie du TP. La banque souhaite proposer un 
nouveau type de compte, le livret ToutBénéf. Dans cette banque, le livret ToutBénéf 
fonctionne comme le compte épargne entreprise. C'est-à-dire qu’il accepte un taux en 
paramètres et applique ce taux au moment de la restitution du solde. 


La première idée qui vient à l’esprit est de créer une classe LivretToutBenef qui hérite 
de CompteEpargneEntreprise. Mais ceci pose un problème si jamais le compte épargne 
entreprise doit évoluer, et c’est justement ce que le directeur de la banque vient de me 
confier. Donc, il vous interdit à juste titre d’hériter de ses fonctionnalités. 


Ce que vous allez donc faire ici, c’est de considérer que le fait qu’un compte puisse faire 
des bénéfices soit en fait un comportement qui est fourni au moment où on instancie un 
compte. Il existe plusieurs comportements dont on doit fournir les implémentations : 


— le comportement de bénéfice à taux fixe ; 
— le comportement de bénéfice aléatoire ; 
— le comportement où il n’y a aucun bénéfice. 


Chaque comportement est une classe qui respecte le contrat suivant : 


1| public interface ICalculateurDeBenefice 
2| € 


3 decimal CalculeBenefice(decimal solde); 


CORRECTION 


4 double Taux { get; } 
Ha 


Écrivez donc ces trois classes de comportement ainsi que le livret ToutBénéf qui possède 
un taux fixe de 2.75% et qui a été crédité une première fois de 800 € et une seconde 
fois de 200 €. Affichez enfin son résumé qui devra tenir compte du taux du calculateur 
de bénéfice. Réécrivez ensuite la classe CompteCourant de manière à ce qu’elle ait un 
comportement où il n’y a pas de bénéfice. Enfin, la classe CompteEpargneEntreprise 
subira une évolution pour fonctionner avec un comportement de bénéfice aléatoire (tiré 
entre 0 et 1). 


C’est parti. 


Correction 


Ici c’est un peu plus compliqué. Vous n'êtes sans doute pas complètement familiers 
avec la notion d'interface, aussi avant de vous donner la correction, je vais vous donner 
quelques pistes. Le fait d’avoir un comportement est finalement très simple. Il suffit 
d’avoir un membre privé dans la classe LivretToutBenef du type de l'interface. Ce 
membre privé sera affecté à la valeur passée en paramètre du constructeur. C’est-à-dire : 


1| public class LivretToutBenef : Compte 

2| € 

3 private ICalculateurDeBenefice calculateurDeBenefice; 

4 

5 public LivretToutBenef (ICalculateurDeBenefice calculateur) 
6 { 

7 calculateurDeBenefice = calculateur; 

8 

o| } 


Désormais, les opérations devront se faire avec cette variable, calculateurDeBenefice. 
On aura également besoin d’instancier au préalable le comportement voulu et de le 
passer en paramètre du constructeur. Retournez donc tenter de réaliser ce TP avant 
de voir la suite de la correction ! 


C’est fait ? Alors voici ma correction. Nous avions donc cette interface imposée : 


1| public interface ICalculateurDeBenefice 

2| { 

3 decimal CalculeBenefice(decimal solde); 
4 double Taux { get; } 

5| } 


Nous avons besoin d'écrire plusieurs classes qui implémentent cette interface. La pre- 
mière est la classe de bénéfice à taux fixe : 


1| public class BeneficeATauxFixe : ICalculateurDeBenefice 
2| € 
3 private double taux; 
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4 

5 public BeneficeATauxFixe (double tauxFixe) 
6 { 

7 taux = tauxFixe; 

8 k 

9 

10 public decimal CalculeBenefice(decimal solde) 
11 { 

12 return solde * (decimal)(1 + taux); 
13 } 

14 

15 public double Taux 

16 { 

17 get 

18 { 

19 return taux; 

20 } 

21 } 

22| } 


Nous avons dit qu’elle devait prendre un taux en paramètre ; le constructeur est l’en- 
droit indiqué pour cela. Ensuite, la méthode de calcul est très simple, il suffit d'appliquer 
la formule au solde. Enfin, la propriété retourne le taux. 


La classe suivante est la classe de bénéfice aléatoire. Là, pas besoin de paramètres, il suf- 
fit de tirer le nombre aléatoire dans le constructeur grâce à la méthode NextDouble(), 
ce qui donne : 


1| public class BeneficeAleatoiïire : ICalculateurDeBenefice 
2| { 

3 private double taux; 

4 private Random random; 

5 

6 public BeneficeAleatoire() 

7 { 

8 random = new Random); 

9 taux = random.NextDouble(); 

10 + 

11 

12 public decimal CalculeBenefice(decimal solde) 
13 { 

14 return solde * (decimal)(1 + taux); 
15 } 

16 

17 public double Taux 

18 { 

19 get 

20 { 

21 return taux; 

22 } 

23 } 


CORRECTION 


24| } 


Enfin, il faudra créer la classe avec aucun bénéfice : 


1| public class AucunBenefice : ICalculateurDeBenefice 
2| € 
3 public decimal CalculeBenefice(decimal solde) 
4 { 
5 return solde; 
6 } 
7 
8 public double Taux 
9 { 
10 get 
11 { 
12 return O; 
13 } 
14 } 


15| } 


Rien de sorcier. 


Là où ça se complique un peu, c’est pour la classe LivretToutBenef. Elle doit bien sûr 
dériver de la classe de base Compte et posséder un membre privé de type 
ICalculateurDeBenefice : 


1| public class LivretToutBenef : Compte 

2| { 

3 private ICalculateurDeBenefice calculateurDeBenefice; 

4 

5 public LivretToutBenef (ICalculateurDeBenefice calculateur) 

6 Re 

T calculateurDeBenefice = calculateur; 

8 2 

9 

10 public override decimal Solde 

11 { 

12 get 

13 { 

14 decimal solde = base.Solde; 

15 return solde + calculateurDeBenefice. 

CalculeBenefice (solde); 

16 } 

17 } 

18 

19 public override void AfficherResume() 

20 { 

21 Console: WriteLine (MPÉTESSRSSSRSARSSS LB 

22 Console.WriteLine("Livret ToutBénéf de " + Proprietaire 
); 

23 Console.WriteLine("\tSolde : " + Solde); 
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24 Console.WriteLine("\tTaux : " + calculateurDeBenefice. 
Taux) ; 

25 AfficheUperations(); 

26 Console:Writeline (NPPÉSESSSRERSÉSRRES LE 

27 } 

28| } 


Dans le constructeur, la variable est initialisée avec un objet de type 
ICalculateurDeBenefice. Ensuite, pour calculer le solde, il suffit d'appeler la mé- 
thode CalculeBenefice avec le solde de base en paramètre. De même, pour faire 
apparaître le taux dans le résumé, on pourra utiliser la propriété Taux de l'interface. 
Il ne reste qu’à souscrire à un Livret ToutBénéf en lui passant un objet de bénéfice à 
taux fixe en paramètre du constructeur, et à faire les opérations bancaires demandées. 
Ce qui donne : 


1| ICalculateurDeBenefice beneficeATauxFixe = new 
BeneficeATauxFixe (0.275); 
2| LivretToutBenef livretToutBenefNicolas = new LivretToutBenef( 


beneficeATauxFixe); 
livretToutBenefNicolas.Crediter (800); 
livretToutBenefNicolas.Crediter (200); 


Console.WriteLine ("Résumé du livret ToutBénéf"); 
livretToutBenefNicolas.AfficherResume() ; 
Console.WriteLine("\n"); 


D I OO Où À 


Vous aurez donc : 
Résumé du livret ToutBénéf 
Livret ToutBénéf de 


Solde : 2275,000 
Taux _: 0,275 


Opérations 
+800 


Maintenant, nous devons adapter la classe CompteEpargneEntreprise pour qu’elle 
puisse fonctionner avec un comportement. Cela devient tout simple et se rapproche 
beaucoup du livret ToutBénéf : 


1| public class CompteEpargneEntreprise : Compte 

2 |: € 

3 private ICalculateurDeBenefice calculateurDeBenefice; 

4 

5 public CompteEpargneEntreprise(ICalculateurDeBenefice 
calculateur) 

6 { 
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T calculateurDeBenefice = calculateur; 

8 } 

9 

10 public override decimal Solde 

11 { 

12 get 

13 

14 decimal solde = base.Solde; 

15 return solde + calculateurDeBenefice. 

CalculeBenefice (solde); 

16 } 

17 } 

19 public override void AfficherResume() 

20 { 

21 Console.WriteLine(" 
HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHN) ; 

22 Console.WriteLine ("Compte épargne entreprise de " + 
Proprietaire); 

23 Console.WriteLine("\tSolde : " + Solde); 

24 Console.WriteLine("\tTaux : " + calculateurDeBenefice. 
Taux) ; 

25 AfficheUperations(); 

26 Console.WriteLine(" 
HHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHHN) ; 

27 x 

28 | } 


Il faut maintenant changer l’instanciation de l’objet CompteEpargneEntreprise en lui 
passant en paramètre un objet de type bénéfice aléatoire : 


1| ICalculateurDeBenefice beneficeAleatoire = new 
BeneficeAleatoire); 

2| CompteEpargneEntreprise compteEpargneNicolas = new 
CompteEpargneEntreprise(beneficeAleatoire) { Proprietaire = 
Nicolas t +: 


Il reste le compte courant qui suit le même principe : 


1| public class CompteCourant : Compte 

2| { 

3 private decimal decouvertAutorise; 

4 private ICalculateurDeBenefice calculateurDeBenefice; 

5 

6 public CompteCourant(decimal decouvert, 
ICalculateurDeBenefice calculateur) 

7 { 

8 decouvertAutorise = decouvert; 

9 calculateurDeBenefice = calculateur; 

10 } 


11 
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12 public override decimal Solde 

13 { 

14 get 

15 { 

16 decimal solde = base.Solde; 

17 return solde + calculateurDeBenefice. 

CalculeBenefice (solde); 

18 } 

19 } 

20 

21 public override void AfficherResume() 

22 { 

23 Console.WriteLine (" 
KO OH DEEE HE OO EE) 5 

24 Console.WriteLine ("Compte courant de " + Proprietaire); 

25 Console.WriteLine ("\tSolde : " + Solde); 

26 Console.WriteLine("\tDécouvert autorisé : " + 
decouvertAutorise); 

CA Console.WriteLine("\tTaux : " + calculateurDeBenefice. 
Taux) ; 

28 AfficheUperations(); 

29 Console.WriteLine (" 
AH HO O KE KO OE  ) 5 

30 } 

31| } 


Que l’on pourra instancier avec un objet de type aucun bénéfice : 


1| ICalculateurDeBenefice aucunBenefice = new AucunBenefice(); 
2| CompteCourant compteNicolas = new CompteCourant (2000, 
aucunBenefice) { Proprietaire = "Nicolas" }; 


Et voilà! L’avantage ici est d’avoir séparé les responsabilités dans différentes classes. 
Si jamais nous créons un nouveau compte qui est rémunéré grâce à un bénéfice à taux 
fixe, il suffira de réutiliser ce comportement et le tour est joué. 


À noter que les trois calculs de la propriété Solde sont identiques, il pourrait être 
judicieux de le factoriser dans la classe mère Compte. Ceci implique que la classe mère 
possède elle-même le membre protégé du type de l'interface. 


Voilà pour ce TP. J'espère que vous aurez réussi avec brio la création de toutes les 
classes et que vous ne vous êtes pas perdus dans les mots-clés. Vous verrez que vous 
aurez très souvent besoin d'écrire des classes dans ce genre afin de créer une application. 
C’est un élément indispensable du C# qu'il est primordial de maîtriser. 


N'hésitez pas à faire des variations sur ce TP ou à créer d’autres petits programmes 
simples vous permettant de vous entraîner ! 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code ) 


Code web : 817413 
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Mode de passage des paramètres à une 
méthode 


Difficulté : 1 


ans les chapitres précédents, nous avons décrit comment on pouvait passer des pa- 

ramètres à une méthode. Nous avons également vu que les types du framework .NET 

pouvaient être des types valeur ou des types référence. Ceci influence la façon dont 
sont traités les paramètres d'une méthode. Nous allons ici préciser un peu ce fonctionne- 
ment. 


Dans ce chapitre, je vais illustrer mes propos en utilisant des méthodes statiques écrites dans 
la classe Program, générée par Visual CH Express. Le but est de simplifier l'écriture et de 
ne pas s'encombrer d'objets inutiles. Évidemment, tout ce que nous allons voir fonctionne 
également de la même façon avec les méthodes non statiques présentes dans des objets. 


265 


CHAPITRE 25. MODE DE PASSAGE DES PARAMÈTRES À UNE MÉTHODE 


Passage de paramètres par valeur 


Au tout début de ce livre, nous avons appelé des méthodes en passant des types simples 
(int, string, etc.). Nous avons vu qu’il s'agissait de types valeur qui possèdent direc- 
tement la valeur dans la variable. 
Lorsque nous passons des types valeur en paramètres d’une méthode, nous utilisons un 
passage de paramètre par valeur. 


Décortiquons l’exemple suivant : 


1| static void Main(string[] args) 
2| € 

3 int age = 30; 

4 Doubler (age) ; 

5 Console .WriteLine (age); 

6| } 

7 

8| public static void Doubler(int valeur) 
| + 

10 valeur = valeur * 2; 

11 Console.WriteLine (valeur); 
12| } 


Nous déclarons dans la méthode Main() une variable age, de type entier, à laquelle 
nous affectons la valeur 30. Nous appelons ensuite la méthode Doubler () en lui passant 
cette variable en paramètre. 


Ce qu’il se passe c’est que le compilateur fait une copie de la valeur de la variable age 
pour la mettre dans la variable valeur de la méthode Doubler(). La variable valeur 
a une portée égale au corps de la méthode Doubler(). 


Nous modifions ensuite la valeur de la variable valeur en la multipliant par deux. 


Étant donné que la variable valeur a reçu une copie de la variable age, c’est-à-dire 
que le compilateur a dupliqué la valeur 30, le fait de modifier la variable valeur ne 
change en rien la valeur de la variable age : 


Nous passons 30 à la méthode Doubler() qui calcule le double de la variable valeur. 
On affiche 60. Lorsqu'on revient dans la méthode Main(), nous retrouvons la valeur 
initiale de la variable age. Elle n’a bien sûr pas été modifiée, car la méthode Doubler () 
a travaillé sur une copie. 


Rappelez-vous que ceci est possible car les types intégrés sont facilement copiables, car 
peu évolués. 
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Passage de paramètres en mise à jour 


Q Oui, mais si je veux modifier la valeur ? 


Pour pouvoir modifier la valeur du paramètre passé, il faut indiquer que la variable 
est en mode « mise à jour ». Cela se fait grâce au mot-clé « ref » que nous pourrons 
utiliser ainsi : 


1| static void Main(string[] args) 
2| € 

3 int age = 30; 

4 Doubler(ref age); 

5 Console.WriteLine (age); 

6| } 

7 

8| public static void Doubler(ref int valeur) 
g| € 

10 valeur = valeur * 2; 

11} } 


Comme on peut s’en douter, ce code affiche 60. 


Le mot-clé ref s'utilise avant la définition du paramètre dans la méthode. Cela implique 
qu’il soit également utilisé au moment d’appeler la méthode. 


Vous aurez remarqué que le mot-clé utilisé est ref et ressemble beaucoup au mot 
« référence ». Ce n’est évidemment pas un hasard! En fait, avec ce mot-clé nous 
demandons au compilateur de passer en paramètre une référence vers la variable age 
plutôt que d’en faire une copie. Ainsi, la méthode Doubler() récupère une référence 
et la variable valeur référence alors la même valeur que la variable age. Ceci implique 
que toute modification de la valeur référencée provoque un changement sur la variable 
source puisque les variables référencent la même valeur. 


Voilà pourquoi la variable est modifiée après le passage dans la méthode Doubler(). 


Bien sûr, il aurait tout à fait été possible de faire en sorte que la méthode Doubler() 
renvoie un entier contenant la valeur passée en paramètre multipliée par 2 : 


1| static void Main(string[] args) 
ARE E 

3 int age = 30; 

4 age = Doubler(age); 

5 Console.WriteLine (age); 

6| } 

7 

8| public static int Doubler(int valeur) 
g|[ € 

10 return valeur * 2; 

11} } 
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C’est ce que nous avons toujours fait auparavant. Voici maintenant une autre façon de 
faire qui peut être bien utile quand il y a plus d’une valeur à renvoyer. 


Passage des objets par référence 


C’est aussi comme ça que cela fonctionne pour les objets. Nous avons vu que les va- 
riables qui stockent des objets possèdent en fait la référence de l’objet. Le fait de passer 
un objet à une méthode équivaut à passer la référence de l’objet en paramètre. Aïnsi, 
c’est comme si on utilisait le mot-clé ref implicitement. 


Le code suivant : 


1| static void Main(string[] args) 

21 { 

3 Voiture voiture = new Voiture { Couleur = "Grise" }; 
4 Repeindre (voiture); 

5 Console.WriteLine (voiture.Couleur); 

6| } 

7 

8| public static void Repeindre (Voiture voiture) 
9! € 

10 voiture.Couleur = "Bleue; 

1]. % 


va donc créer un objet Voiture et le passer en paramètre à la méthode Repeindre(). 
Comme Voiture est un type référence, il n’y a pas de duplication de l’objet et une 
référence est passée à la méthode. 


Lorsque nous modifions la propriété Couleur de la voiture, nous modifions bien le 
même objet que celui présent dans la méthode Main() : 


Il est à noter quand même que la variable voiture de la méthode Repeindre est une 
copie de la variable voiture de la méthode Main(). Elles contiennent toutes les deux 
une référence vers l’objet de type Voiture. Cela veut dire que l’on accède bien au 
même objet, d’où le résultat, mais que les deux variables sont indépendantes. Si nous 
modifions directement la variable, avec par exemple : 


1| static void Main(string[] args) 

à let 

3 Voiture voiture = new Voiture { Couleur = "Grise" }; 
4 Repeindre (voiture); 

5 Console.WriteLine (voiture.Couleur); 

6| } 

7 

8| public static void Repeindre (Voiture voiture) 

9! € 

10 voiture.Couleur = "Bleue; 
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11 voiture = null; 


12| } 


alors, ce code continuera à fonctionner, car la variable voiture de la méthode Main() 
ne vaut pas null. Le fait de modifier la variable de la méthode Repeindre, qui est une 
copie, n’affecte en rien la variable de la méthode Main(). 


Par contre, elle est affectée si nous utilisons le mot-clé ref. Par exemple le code suivant : 


1| static void Main(string[] args) 

2! { 

3 Voiture voiture = new Voiture { Couleur = "Grise" #}; 
4 Repeindre(ref voiture); 

5 Console.WriteLine (voiture.Couleur); 

6| } 

vd 

8| public static void Repeindre(ref Voiture voiture) 
9! { 

10 voiture.Couleur = "Bleue"; 

11 voiture = null; 

121% 


provoquera une erreur. En effet, cette fois-ci, c’est bien la référence qui est passée à 
nulle et pas une copie de la variable contenant la référence... 


Une différence subtile ! 


Passage de paramètres en sortie 


Enfin, le dernier mode de passage est le passage de paramètres en sortie. Il permet 
de faire en sorte qu’une méthode force l’initialisation d’une variable et que l’appelant 
récupère la valeur initialisée. Nous avons déjà vu ce mode de passage quand nous avons 
étudié les conversions. Souvenez-vous de ce code : 


string chaine = "1234"; 

int nombre; 

if (int.TryParse(chaine, out nombre)) 
Console.WriteLine (nombre); 

else 


Oo Où À © ND 


Console.WriteLine ("Conversion impossible"); 


La méthode TryParse permet de tenter la conversion d’une chaîne. Elle renvoie vrai 
ou faux en fonction du résultat de la conversion et met à jour l’entier qui est passé 
en paramètre en utilisant le mot-clé out. Si la conversion réussit, alors l’entier nombre 
est initialisé avec la valeur de la conversion, calculée dans la méthode TryParse. Nous 
pouvons utiliser ensuite la variable nombre car le mot-clé out garantit que la variable 
sera initialisée dans la méthode. 


En effet, si nous prenons l’exemple suivant : 
4 


1 | static void Main(string[] args) 
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2| { 

3 int age = 30; 

4 int ageDouble; 

5 Doubler (age, out ageDouble); 

6| } 

7 

8| public static void Doubler(int age, out int resultat) 
o| { 

10] } 


Nous aurons une erreur de compilation : 


Le paramètre de sortie resultat” doit être assigné avant que le 


contrôle quitte la méthode actuelle 


En effet, il faut absolument que la variable resultat qui est marquée en sortie ait une 
valeur avant de pouvoir sortir de la méthode. 


Nous pourrons corriger cet exemple avec : 


1| static void Main(string[] args) 
2| { 

3 int age = 30; 

4 int ageDouble; 

5 Doubler(age, out ageDouble); 
6| } 

7 

8| public static void Doubler(int age, out int resultat) 
o| { 

10 resultat = age * 2; 

ul} 


Après l’appel de la méthode Doubler(), ageDouble vaudra donc 60. 


Oui, mais quel intérêt par rapport à un return normal ? 


Ici, aucun. C’est pertinent quand nous souhaitons renvoyer plusieurs valeurs, comme 
c’est le cas pour la méthode TryParse qui renvoie le résultat de la conversion et si la 
conversion s’est bien passée. 


Bien sûr, si l’on n’aime pas trop le mot-clé out, il est toujours possible de créer un 
objet contenant deux valeurs que l’on retournera à l’appelant, par exemple : 


1| public class Program 

2| € 

3 static void Main(string[]l args) 

4 { 

5 string nombre = "1234"; 

6 Resultat resultat = TryParse (nombre); 
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7 if (resultat.ConversionOk) 

8 Console.WriteLine (resultat.Valeur); 

9 } 

10 

11 public static Resultat TryParse(string chaine) 

12 { 

13 Resultat resultat = new Resultat (); 

14 int valeur; 

15 resultat.ConversionOk = int.TryParse(chaine, out valeur 
); 

16 resultat.Valeur = valeur; 

17 return resultat; 

18 } 

19 

20 | } 

21 

22| public class Resultat 

23 | { 

24 public bool Conversion0k { get; set; } 

25 public int Valeur { get; set; } 

26| } 


Ici, notre méthode TryParse renvoie un objet Resultat qui contient les deux valeurs 
résultantes de la conversion. 


En résumé 


— Le type d’une variable passée en paramètres d’une méthode influence la façon dont 
elle est traitée. 

— Un passage par valeur effectue une copie de la valeur de la variable et la met dans 
la variable de la méthode. 

— Un passage par référence effectue une copie de la référence mais continue de pointer 
sur le même objet. 

— On utilise le mot-clé ref pour passer une variable de type valeur à une méthode afin 
de la modifier. 
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Les structures 


Difficulté : 16: 


ous allons aborder dans ce chapitre les structures, qui sont une nouvelle sorte d'ob- 

jets que nous pouvons créer dans des applications C#. Les structures sont presque 

comme des classes. Elles permettent également de créer des objets, possèdent des 
variables ou propriétés, des méthodes et même un constructeur, mais avec quelques subtiles 
différences... 


Découvrons-les dès à présent ! 
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Une structure est presque une classe 


Une des premières différences entre la classe et la structure est la façon dont .NET 
gère ces deux objets. Nous avons vu que les classes étaient des types référence. Les 
variables de type référence ne possèdent donc pas la valeur de l’objet mais une référence 
vers cet objet en mémoire. La structure, quant à elle, est un type valeur et contient 
donc directement la valeur de l’objet. 


Une autre différence est que la structure, bien qu’étant un objet, ne peut pas utiliser 
les principes d’héritage. On ne peut donc pas hériter d’une structure et une structure 
ne peut pas hériter des comportements d’un objet. 


À quoi sert une structure ? 


Les structures vont être utiles pour stocker de petits objets amenés à être souvent 
manipulés, comme les int ou les bool que nous avons déjà vus. 


La raison tient en un seul mot : performance. 


Étant gérées en mémoire différemment, les structures sont optimisées pour améliorer les 
performances des petits objets. Comme il n’y a pas de référence, on utilisera directement 
l’objet sans aller le chercher via sa référence. On gagne donc un peu de temps lorsqu'on 
a besoin de manipuler ces données. 


C’est tout à fait pertinent pour des programmes où la vitesse est déterminante, comme 
les jeux vidéo. 


Q Donc dans ce cas, autant utiliser tout le temps des structures, non ? 


Eh bien non, déjà vous vous priveriez de tous les mécanismes d’héritage que nous avons 
vus. Ensuite, si nous surchargeons trop la mémoire avec des structures, l’optimisation 
prévue par .NET risque de se retourner contre nous et notre application pourrait être 
plus lente que si nous avions utilisé des classes. 

D'une manière générale, et à moins de savoir exactement ce que vous faites ou d’avoir 
mesuré les performances, vous allez utiliser plus généralement les classes que les struc- 
tures. Vous pouvez à ce sujet lire les recommandations de Microsoft, disponibles grâce 
au code web suivant : 


Documentation Microsoft ) 


Code web : 520338 
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Créer une structure 


Pour créer une structure, nous utiliserons le mot-clé struct, comme nous avions utilisé 
le mot-clé class pour créer une classe : 


1| public struct Personne 

2| € 

3 public string Prenom { get; set; } 
4 public int Age { get; set; } 

5| } 


Pour instancier cette structure, nous pourrons utiliser le mot-clé new, comme pour les 
classes. La différence est que la variable sera un type valeur, avec les conséquences que 
ce type impose en matière de gestion en mémoire ou de passages par paramètres : 


1| Personne nicolas = new Personne() { Prenom = "Nicolas", Age = 
30 }; 

2| Console.WriteLine(nicolas.Prenom + " a " + nicolas.Age + " ans" 
); 


Comme nous avons dit, il est impossible qu’une structure hérite d’une autre structure ou 
d’un objet ; sauf bien sûr du fameux type de base object, pour qui c’est automatique. 
Une structure hérite donc des quelques méthodes d’Object (comme ToString()) que 
nous pouvons éventuellement spécialiser : 


1| public struct Personne 

2| € 

3 public string Prenom { get; set; } 

4 public int Age { get; set; } 

5 

6 public override string ToString() 

7 { 

8 return Prenom + " a " + Age + " ans"; 
9 } 

10| } 


Et nous pourrons écrire : 
1| Personne nicolas = new Personne () { Prenom = "Nicolas", Age = 


30: xs 
2| Console.WriteLine(nicolas.ToString()); 


Qui renverra : 


Nicolas a 30 ans 


Comme pour les classes, il est possible d’avoir des constructeurs sur une structure, à 
l'exception du constructeur par défaut qui est interdit. 


Aussi le code suivant : 
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1| public struct Personne 
2| € 

3 public Personne() 
4 { 

5 } 

6| } 


provoquera l'erreur de compilation suivante : 


Les structures ne peuvent pas contenir de constructeurs exempts 


de paramètres explicites 


Par contre, les autres formes des constructeurs sont possibles, comme : 


1| public struct Personne 

2| € 

3 private int age; 

4 public Personne(int agePersonne) 
5 { 

6 age = agePersonne; 

7 } 

8| } 


Ce constructeur s’utilisera comme pour une classe : 


1 | Personne nicolas = new Personne (30); 


Attention, si vous tentez d'utiliser des propriétés ou des méthodes dans le 
constructeur d'une structure, vous allez avoir un problème! 


Par exemple le code suivant : 


1| public struct Personne 

2| { 

3 private int age; 

4 public Personne(int agePersonne) 
5 { 

6 AffecteAge (agePersonne); 

7 } 

8 

9 private void AffecteAge(int agePersonne) 
10 { 

11 age = agePersonne; 

12 } 

13| } 


provoquera les erreurs de compilation suivantes : 


L’objet ?’this’ ne peut pas être utilisé avant que tous ses 
champs soient assignés 
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Le champ ?’MaPremiereApplication.Personne.age” doit être 
totalement assigné avant que le contrôle soit retourné à 1? 
appelant 


Alors qu'avec une classe, ce code serait tout à fait correct. 


Pour corriger ceci, il faut absolument initialiser tous les champs avant de faire quoi que 
ce soit avec l’objet, comme l'indique l’erreur. 


Nous pourrons par exemple faire : 


1| public struct Personne 

2 NT 

3 private int age; 

4 public Personne(int agePersonne) 
5 L 

6 age = 0; 

7 AffecteAge (agePersonne); 

8 } 

9 

10 private void AffecteAge(int agePersonne) 
11 { 

12 age = agePersonne; 

13 } 

14/| } 


Ce qui peut sembler tout à fait inutile dans ce cas-là. Mais comme le compilateur fait 
certaines vérifications, il sera impossible de compiler un code de ce genre sans que 
toutes les variables soient initialisées explicitement. 


Par contre, vous aurez un souci si vous utilisez des propriétés automatiques. Si nous 
tentons de faire : 


1| public struct Personne 

2| € 

3 public int Age { get; set; } 

4 public Personne(int agePersonne) 
5 { 

6 Age = agePersonne; 

7 } 

8| } 


nous nous retrouverons avec la même erreur de compilation. Pour la corriger, il faudra 
appeler le constructeur par défaut de la structure qui va permettre d’initialiser toutes 
les variables de la classe : 


1| public struct Personne 

2| € 

3 public int Age { get; set; } 

4 public Personne(int agePersonne) : this() 
5 { 

6 Age = agePersonne; 
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x 
Cela se fait comme pour les classes, en utilisant le mot-clé this suivi de parenthèses, 
qui permettra d'appeler le constructeur par défaut. 


Rappelez-vous que le constructeur par défaut s'occupe d’initialiser toutes les variables 
d’une classe ou d’une structure. 


Passage de structures en paramètres 


Comme il s’agit d’un type valeur, à chaque fois que nous passerons une structure en 
paramètres d’une méthode, une copie de l’objet sera faite. 


Ainsi, tout à fait logiquement, le code suivant : 


1| static void Main(string[] args) 

2| € 

3 Personne nicolas = new Personne() { Age = 30 }; 
4 FaitVieillir(nicolas); 

5 Console.WriteLine(nicolas.Age); 

6| } 

7 

8| private static void FaitVieillir (Personne personne) 
o| + 

10 personne .Age++; 

ul} 


affichera 30, bien que nous modifions l’âge de la personne dans la méthode. 
Comme nous l’avons déjà vu, la méthode travaille sur une copie de la structure. 


Cela veut bien sûr dire que si nous souhaitons modifier une structure à partir d’une 
méthode, nous devrons utiliser le mot-clé ref : 


1| static void Main(string[] args) 

2| € 

3 Personne nicolas = new Personne() { Age = 30 }; 
4 FaitVieillir(ref nicolas); 

5 Console.WriteLine(nicolas.Age); 

6| } 

7 

8| private static void FaitVieillir(ref Personne personne) 
o| + 

10 personne .Age++; 

ul} 


Ceci vaut pour tous les types valeur. 


Prenez quand même garde. Si la structure est très grosse, le fait d'en faire une 
copie à chaque utilisation de méthode risque d'être particulièrement chrono- 
phage et pourra perturber les performances de votre application. 
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D’autres structures ? 


Vous l’aurez peut-être deviné, mais les entiers que nous avons déjà vus et que nous 
utilisions grâce au mot-clé int sont en fait des structures. 


Étant très souvent utilisés et n’ayant pas non plus énormément de choses à stocker, ils 
sont créés en tant que structures et sont optimisés par .NET pour que nos applications 
s’exécutent de façon optimale ; ce qui est un choix tout à fait pertinent. 


C’est le cas également pour les bool, les double, etc. 


À noter en revanche que d’autres objets, comme la classe String, sont bel et bien des 
classes. 


D'une manière générale, vous allez créer peu de structures en tant que débutant. Il 
sera plus judicieux de créer des classes dès que vous en avez besoin. En effet, plus vos 
objets sont gros et plus ils auront intérêt à être des classes pour éviter d’être recopiés 
à chaque utilisation. 


L'utilisation de structures pourra se révéler pertinente dans des situations bien précises, 
mais en général, il faut bien maîtriser les rouages du framework .NET pour que les 
bénéfices de leur utilisation se fassent ressentir. 


Dans tous les cas, il sera important de mesurer (avec par exemple des outils de profilage) 
le gain de temps avant de mettre des structures partout. 


En résumé 


— Les structures sont des types valeur qui sont optimisés par le framework .NET. 

— Une structure est un objet qui ressemble beaucoup à une classe, mais qui possède 
des restrictions. 

— Les structures possèdent des propriétés, des méthodes, des constructeurs, etc. 

— Il n’est pas possible d'utiliser l'héritage avec les structures. 
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Re RE 


Les génériques 


Difficulté : 5:53 


es génériques sont une fonctionnalité du framework .NET apparus avec la version 

2. Vous vous en souvenez peut-être, nous avons cité le mot dans le chapitre sur 

les tableaux et les listes. Ils permettent de créer des méthodes ou des classes qui 
sont indépendantes d'un type. Il est très important de connaître leur fonctionnement car 
c'est un mécanisme clé qui permet de faire beaucoup de choses, notamment en termes de 
réutilisabilité et d'amélioration des performances. 


N'oubliez pas vos tubes d'aspirine et voyons à présent de quoi il retourne! 
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Qu'est-ce que les génériques ? 


Avec les génériques, vous pouvez créer des méthodes ou des classes qui sont indépen- 
dantes d’un type. On les appellera des méthodes génériques et des types génériques. 


Nous en avons déjà utilisé, rappelez-vous, avec la liste. 


La liste est une classe comme nous en avons déjà vu plein, sauf qu’elle a la capa- 
cité d’être utilisée avec n'importe quel autre type, comme les entiers, les chaînes de 
caractères, les voitures... 


Cela permet d’éviter de devoir créer une classe Listelnt, une classe ListeString, une 
classe ListeVoiture, etc. On pourra utiliser cette classe avec tous les types grâce aux 
chevrons : 


1| List<string> listeChaine = new List<string>(); 
2| List<int> listeEntier = new List<int>(); 
3| List<Voiture> listeVoiture = new List<Voiture>(); 


Nous indiquons entre les chevrons le type qui sera utilisé avec le type générique. 


liste, il suffirait de créer une ListeObject ? Puisque tous les objets dérivent 


Q Oui mais, si nous voulons pouvoir mettre n'importe quel type d'objet dans une 
d'object... 


En fait, c’est le choix qui avait été fait dans la toute première version de .NET. On 
utilisait l’objet ArrayList qui possède une méthode Add prenant en paramètre un 
object. Cela fonctionne. Sauf que nous nous trouvions face à des limitations. 


Premièrement, nous pouvions mélanger n’importe quel type d’objet dans la liste, des 
entiers, des voitures, des chiens, etc. Cela devenait une classe fourre-tout et nous ne 
savions jamais ce qu'il y avait dans la liste. 


Deuxièmement, même si nous savions qu'il n’y avait que des entiers dans la liste, nous 
étions obligés de les traiter en tant qu'object et donc d'utiliser le boxing et l’unboxing 
pour mettre les objets dans la liste ou pour les récupérer. 


Cela engendrait donc confusion et perte de performance. Grâce aux génériques, il deve- 
nait donc possible de créer des listes de n’importe quel type avec la garantie de savoir 
exactement quel type nous allions récupérer dans la liste. 


Les types génériques du framework .NET 


Le framework .NET possède beaucoup de classes et d’interfaces génériques, notamment 
dans l’espace de nom System.Collections.Generic. 


La liste est la classe générique que vous utiliserez sûrement le plus. Mais beaucoup 
d’autres sont à votre disposition. Citons par exemple la classe Queue<> qui permet de 
gérer une file d’attente style FIFO ! : 


1. Acronyme de l'expression anglaise First In, First Out : premier entré, premier sorti. 
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Queue<int> file = new Queue<int>(); 
file.Enqueue(3); 

file.Enqueue(1); 

int valeur = file.Dequeue(); // valeur contient 3 
valeur = file.Dequeue(); // valeur contient 1 


Oo & © ND 


Citons encore le dictionnaire d’éléments qui est une espèce d'annuaire où l’on accède 
aux éléments grâce à une clé : 


1] Dictionary<string, Personne> annuaire = new Dictionary<string, 
Personne >() ; 

2| annuaire.Add("06 01 02 03 04", new Personne { Prenom = "Nicolas 
JS 

3| annuaire.Add("06 06 06 06 06", new Personne { Prenom = "Jeremie 
W +); 

4 

5| Personne p = annuaire["06 06 06 06 06"]; // p contient la 


propriété Prenom valant Jeremie 


Loin de moi l’idée de vous énumérer toutes les collections génériques du framework 
.NET ; le but est de vous montrer rapidement qu’il existe beaucoup de classes génériques 
dans le framework .NET. 


Créer une méthode générique 


Nous commençons à cerner l’intérêt des génériques. Sachez qu’il est bien sûr possible 
de créer vos propres classes génériques ou vos propres méthodes. 


Commençons par les méthodes, ça sera plus simple. I est globalement intéressant 
d'utiliser un type générique partout où nous pourrions avoir un object. 


Dans la première partie, nous avions créé une méthode AfficheRepresentation() qui 
prenait un objet en paramètre, ce qui pourrait être : 


1| public static class Afficheur 
2| { 
3 public static void Affiche(object o) 
4 { 
5 Console.WriteLine("Afficheur d'objet :"); 
6 Console.WriteLine("\tType : " + o.GetType()); 
7 Console.WriteLine("\tReprésentation : " + o.ToString()) 
8 } 
} 


Nous avons ici utilisé une classe statique permettant d’afficher le type d’un objet et sa 
représentation. Nous pouvons l'utiliser ainsi : 


1| int i = B&; 
2| double d = 9.5; 
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string s = "abcd'; 
Voiture v = new Voiture); 


Afficheur.Affiche(i); 
Afficheur.Affiche(d); 
Afficheur.Affiche(s); 
Afficheur.Affiche(v); 


© OO 1 OO O1 À 


Rappelez-vous, chaque fois qu’on passe dans cette méthode, l’objet est boxé en type 
object quand il s’agit d’un type valeur. 


Nous pouvons améliorer cette méthode en créant une méthode générique. Regardons 
ce code : 


1| public static class Afficheur 

2| € 

3 public static void Affiche<T>(T a) 

4 { 

5 Console.WriteLine ("Afficheur d'objet :"); 

6 Console.WriteLine("\tType : " + a.GetType()); 

7 Console.WriteLine("\tReprésentation : " + a.ToString()) 
8 } 

o[} 


Cette méthode fait exactement la même chose mais avec les génériques. 


Dans un premier temps, la méthode annonce qu’elle va utiliser un type générique 
représenté par la lettre « T » entre chevrons. 


Il est conventionnel que les types génériques soient utilisés avec « T ». 


Cela signifie que tout type utilisé dans cette méthode, déclaré avec T, sera du type 
passé à la méthode. Ainsi, la variable a est du type générique qui sera précisé lors 
de l’appel à cette méthode. Comme a est un objet, nous pouvons appeler la méthode 
GetType() et la méthode ToString() sur cet objet. 


Pour afficher un objet, nous pourrons faire : 


int i = 5; 

double d = 9.5; 

string s = "abcd'; 

Voiture v = new Voiture); 


Afficheur.Affiche<int>(i); 

Afficheur.Affiche <double>(d); 
Afficheur .Affiche<string>(s); 
Afficheur.Affiche<Voiture>(v); 


© ® I OO OÙ À À NN H 
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Dans le premier appel, nous indiquons que nous souhaitons afficher i dont le type 
générique est int. Tout se passe comme si le CLR créait la surcharge de la méthode 
Affiche, prenant un entier en paramètre : 


1| public static void Affiche(int a) 

2| € 

3 Console.WriteLine ("Afficheur d'objet :"); 

4 Console.WriteLine("\tType : " + a.GetType()); 

5 Console.WriteLine("\tReprésentation : " + a.ToString()); 
6| } 


De même pour l'affichage suivant, où l’on indique le type double entre les chevrons. 
C’est comme si le CLR créait la surcharge prenant un double en paramètre : 


1| public static void Affiche (double a) 

2| € 

3 Console.WriteLine ("Afficheur d'objet :"); 

4 Console.WriteLine("\tType : " + a.GetType()); 

5 Console.WriteLine("\tReprésentation : " + a.ToString()); 
6| } 


Et ceci pour tous les types utilisés, à savoir ici int, double, string et Voiture. 


À noter que dans cet exemple, nous pouvons remplacer les quatre lignes suivantes : 


Afficheur.Affiche<int>(i); 

Afficheur.Affiche <double >(d) ; 
Afficheur.Affiche<string>(s); 
Afficheur.Affiche<Voiture>(v); 


& CU ND 


par : 


Afficheur.Affiche(i); 
Afficheur.Affiche(d); 
Afficheur.Affiche(s); 
Afficheur.Affiche(v); 


& © NN 


En effet, nul besoin de préciser quel type nous souhaitons traiter ici, le compilateur est 
assez malin pour le déduire du type de la variable. La variable i étant un entier, il est 
obligatoire que le type générique soit un entier. Il est donc facultatif ici de le préciser. 


Une fois qu'il est précisé entre les chevrons, le type générique s’utilise dans la méthode 
comme n'importe quel autre type. Nous pouvons avoir autant de paramètres génériques 
que nous le voulons dans les paramètres et utiliser le type générique dans le corps de 
la méthode. Par exemple, la méthode suivante : 


1| public static void Echanger <T>(ref T t1, ref T t2) 
2| € 

3 T temp = ti; 

4 EL = 2; 

5 t2 = temp; 

6| } 


285 


CHAPITRE 27. LES GÉNÉRIQUES 


permet d'échanger le contenu de deux variables entre elles. C’est donc une méthode 
générique puisqu'elle précise entre les chevrons que nous pourrons utiliser le type T. 


En paramètres de la méthode, nous passons deux variables de types génériques. 


Dans le corps de la méthode, nous créons une variable du type générique qui sert 
de mémoire temporaire puis nous échangeons les références des deux variables. Nous 
pourrons utiliser cette méthode ainsi : 


int i = 5; 

int j = 10; 
Echanger(ref i, ref j); 
Console.WriteLine(i); 
Console.WriteLine(j); 


Voiture vi new Voiture { Couleur "Rouge" }; 
Voiture v2 = new Voiture { Couleur = "Verte" }; 
Echanger(ref vi, ref v2); 
Console.WriteLine(vi.Couleur) ; 


Console.WriteLine(v2.Couleur) ; 


© © 1 OO OÙ À À ND H 


+= 
© 


11 


Qui donnera : 


Il est bien sûr possible de créer des méthodes prenant en paramètres plusieurs types gé- 
nériques différents. Il suffit alors de préciser autant de types différents entre les chevrons 
qu'il y a de types génériques différents : 


1| static void Main(string[] args) 

2| { 

3 int i = 5; 

4 int jy = 5; 

5 double d = 9.5; 

6 

7 Console.WriteLine(EstEgal(i, j)); 
8 Console.WriteLine (EstEgal(i, d)); 
9} 

10] public static bool EstEgal<T, U>(T t, U u) 
11] À 

12 return t.Equals(u); 

131} 


Ici, la methode EstEgal () prend en paramètres deux types potentiellement différents. 
Nous l’appelons une première fois avec deux entiers et ensuite avec un entier et un 


double. 
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Créer une classe générique 


Une classe générique fonctionne comme pour les méthodes. C’est une classe où nous 
pouvons indiquer de 1 à N types génériques. C’est comme cela que fonctionne la liste 
que nous avons déjà beaucoup manipulée. 


En fait, la liste n’est qu’une espèce de tableau évolué. Nous pourrions très bien imaginer 
créer notre propre liste sur ce principe, sachant que c’est complètement absurde, car 
elle sera forcément moins bien que cette classe, mais c’est pour l'étude. 


Le principe est d’avoir un tableau plus ou moins dynamique qui grossit si jamais le 
nombre d'éléments devient trop grand pour sa capacité. 


Pour déclarer une classe générique, nous utiliserons à nouveau les chevrons à la fin de 
la ligne qui déclare la classe : 


1| public class MaListeGenerique <T> 
2| € 
3| } 


Nous allons réaliser une implémentation toute basique de cette classe histoire de voir un 
peu à quoi ressemble une classe générique. Cette classe n’a d’intérêt que pour étudier 
les génériques, vous lui préférerez évidemment la classe List<> du framework .NET. 


Nous avons besoin de trois variables privées. La capacité de la liste, le nombre d’élé- 
ments dans la liste et le tableau générique. 


1| public class MaListeGenerique <T> 
2| € 

3 private int capacite; 

4 private int nbElements; 

5 private T[] tableau; 

6 

7 public MaListeGenerique() 

8 { 

9 capacite = 10; 

10 nbElements = O0; 

11 tableau = new T[capacitel]l; 
12 } 

13| } 


Remarquez la déclaration du tableau. Elle utilise le type générique. Concrètement, cela 
veut dire que quand nous utiliserons la liste avec un entier, nous aurons un tableau 
d’entiers. Lorsque nous utiliserons la liste avec un objet Voiture, nous aurons un 
tableau de Voiture, etc. 


Nous initialisons ces variables membres dans le constructeur, en décidant complètement 
arbitrairement que la capacité par défaut de notre liste est de 10 éléments. La dernière 
ligne instancie le tableau en lui indiquant sa taille. 


Il reste à implémenter l’ajout dans la liste : 


1 | public class MaListeGenerique <T> 
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2| € 

3 [Code enlevé pour plus de clarté] 

4 

5 public void Ajouter (T element) 

6 { 

7 if (nbElements >= capacite) 

8 { 

9 capacite %*= 2; 

10 T[] copieTableau = new T[capacitel; 
11 for (int i = 0; i < tableau.Length; i++) 
12 { 

13 copieTableaul[i] = tableaulil]; 
14 } 

15 tableau = copieTableau; 

16 } 

17 tableau[nbElements] = element; 

18 nbElements++; 

19 } 

20| } 


Il s’agit simplement de mettre la valeur que l’on souhaite ajouter à l'emplacement adé- 
quat dans le tableau. Nous le mettons en dernière position, c’est-à-dire à l'emplacement 
correspondant au nombre d'éléments. 


Au début, nous avons commencé par vérifier si le nombre d’éléments était supérieur 
à la capacité du tableau. Si c’est le cas, alors nous devons augmenter la capacité du 
tableau. J’ai ici décidé encore complètement arbitrairement que je doublais la capacité. 
Il ne reste plus qu’à créer un nouveau tableau de cette nouvelle capacité et à copier les 
éléments du premier tableau dans celui-ci. 


Vous aurez noté que le paramètre de la méthode Ajouter est bien du type générique. 


Pour le plaisir, rajoutons enfin une méthode permettant de récupérer un élément d’in- 
dice donné : 


1| public class MaListeGenerique <T> 

2| € 

3 [Code enlevé pour plus de clarté] 

4 

5 public T ObtenirElement(int indice) 
6 { 

7 return tableaulindicel; 

8 } 

o[} 


Il s’agit juste d'accéder au tableau pour renvoyer la valeur à l’indice concerné. L'élément 
intéressant ici est de constater que le type de retour de la méthode est bien du type 
générique. 

Cette liste peut s’utiliser de la manière suivante : 


1| MaListeGenerique<int> maListe = new MaListeGenerique<int>(); 
2| maListe.Ajouter (25); 
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maListe.Ajouter (30); 
maListe.Ajouter (5); 


Console.WriteLine (maListe.0ObtenirElement (1)); 


3 
4 
5 
6| Console.WriteLine(maListe.0btenirElement (0)); 
7 
8| Console.WriteLine(maListe.0ObtenirElement (2)); 
9 


10|[ for (int i = O0; i < 30; i++) 


11| À 
12 maListe.Ajouter(i); 
13|[ } 


Ici, nous utilisons la liste avec un entier, mais elle fonctionnerait tout aussi bien avec un 
autre type. N'hésitez pas à passer en debug dans la méthode Ajouter() pour observer 
ce qui se passe exactement lors de l’augmentation de capacité. 


Voilà comment on crée une classe générique ! 


Une fois qu’on à compris que le type générique s’utilise comme n'importe quel autre 
type, cela devient assez facile. 


Rappelez-vous, toute classe qui manipule des object est susceptible d'être 
améliorée en utilisant les génériques. 


La valeur par défaut d’un type générique 


Vous aurez remarqué dans l’implémentation de la liste du dessus que si nous essayons 
d'obtenir un élément du tableau à un indice qui n’existe pas, nous aurons une erreur. 
Ce comportement est une bonne chose, car il est important de gérer les cas aux limites. 
En l’occurrence ici, on délègue au tableau la gestion du cas limite. 


On pourrait envisager de gérer nous-mêmes ce cas limite en affichant un message et en 
renvoyant une valeur nulle, mais ceci pose un problème. Pour un objet Voiture, qui 
est un type référence, il est tout à fait pertinent d’avoir null; pour un int, qui est 
un type valeur, ça n’a pas de sens. C’est là qu’intervient le mot-clé default. Comme 
son nom l'indique, il renvoie la valeur par défaut du type. Pour un type référence, c’est 
null ; pour un type valeur c’est 0. Ce qui donnerait : 


1| public T ObtenirElement (int indice) 

2! { 

3 if (indice < O || indice >= nbElements) 

4 { 

5 Console.WriteLine("L'indice n'est pas bon"); 
6 return default (T); 

7 } 

8 return tableaulindicel; 

91} 
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Les interfaces génériques 


Les interfaces peuvent aussi être génériques. D'ailleurs, ça me fait penser que plus 
haut, je vous ai indiqué qu’un certain code n'était pas très esthétique et que j'en 
parlerai plus tard... Le moment est venu! Vous ne vous en souvenez plus ? Petit rappel 
pour les étourdis : il s'agissait du chapitre sur les interfaces où nous avions implémenté 
l'interface IComparable. 


Nous souhaitions comparer des voitures entre elles et nous avions obtenu le code sui- 
vant : 


1| public class Voiture : IComparable 

2| € 

3 public string Couleur { get; set; } 
4 public string Marque { get; set; } 
5 public int Vitesse { get; set; } 
6 

7 

8 

9 


public int CompareTo(object obj) 
{ 
Voiture voiture = (Voiture)obj; 
10 return Vitesse.CompareTo(voiture.Vitesse); 
11 F 
12] } 


Je souhaite pouvoir comparer des voitures entre elles, mais le framework .NET me 
fournit un object en paramètres de la méthode CompareTo(). Quelle idée! Comme si 
je voulais comparer des voitures avec des chats ou des chiens. Cela me force en plus à 
faire un cast. Pourquoi il ne me passe pas directement une Voiture en paramètre ? 


Vous en avez l'intuition ? Un object! Oui, mais c’est un peu lourd à manier... et je 
ne parle pas des performances ! 


C’est là où les génériques vont voler à notre secours. L'interface IComparable date 
de la première version du framework .NET. Le C# ne possédait pas encore les types 
génériques. Depuis leur apparition, il est possible d’implémenter la version générique 
de cette interface. 


Pour cela, nous faisons suivre l’interface du type que nous souhaitons utiliser entre les 
chevrons. Cela donne : 


1| public class Voiture : IComparable <Voiture > 
2| € 

3 public string Couleur { get; set; } 

4 public string Marque { get; set; } 

5 public int Vitesse { get; set; } 
6 

7 

8 

9 


public int CompareTo(Voiture obj) 
{ 
return Vitesse.CompareTo(obj.Vitesse); 
10 } 
ul} 


LES RESTRICTIONS SUR LES TYPES GÉNÉRIQUES 


Nous devons toujours implémenter la méthode CompareTo() sauf que nous avons dé- 


sormais un objet Voiture en paramètre, ce qui nous évite de le caster. 


Les restrictions sur les types génériques 


Une méthode ou une classe générique c’est bien, mais peut-être voulons-nous qu’elles 
ne fonctionnent pas avec tous les types. Aussi, le C;£ permet de définir des restrictions 


sur les types génériques. Pour ce faire, on utilise le mot-clé where. 


Par exemple, nous pouvons définir une restriction sur une méthode générique afin 


Il existe six types de restrictions : 


Contrainte 


Description 


where T : struct 


Le type générique doit être un type valeur 


where T : class 


Le type générique doit être un type référence 


where T : new() 


Le type générique doit posséder un constructeur par 
défaut 


where T : IMonInterface 


Le type générique doit implémenter l'interface 
IMonlnterface 


where T : MaClasse 


Le type générique doit dériver de la classe MaClasse 


where T1 : T2 


Le type générique doit dériver du type générique T2 


qu’elle n’accepte en type générique que des types qui implémentent une interface. 


Soit l'interface suivante : 


1 
2 
3 
4 
5 


public interface IVolant 


{ 


void DeplierLesAiles(); 


void Voler(); 
} 


qui est implémentée par deux objets : 


1 
2 
3 
4 
5 
6 
7 
8 
9 


public class Avion 


{ 


IVolant 


public void DeplierLesAiles() 


€ 


Console.WriteLine ("Je déplie mes ailes mécaniques"); 


F 


public void Voler() 


{ 


Console.WriteLine("J'allume le moteur"); 


} 


public class Üiseau 


{ 


IVolant 
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16 public void DeplierLesAiles() 

17 { 

18 Console.WriteLine("Je déplie mes ailes d'oiseau"); 
19 } 

20 

21 public void Voler() 

22 { 

23 Console.WriteLine("Je bas des ailes"); 

24 } 

25| } 


Nous pouvons créer une méthode générique qui s’occupe d’instancier ces objets et 
d’appeler les méthodes de l'interface : 


1| public static T Creer<T>() where T : IVolant, new() 
2| À 

3 T t = new T(); 

4 t.DeplierLesAiles(); 

5 t.Voler(); 

6 return &:; 

7|} 


Ici, la restriction porte sur deux niveaux. Il faut dans un premier temps que le type gé- 
nérique implémente l'interface IVolant. En outre, il faut qu’il possède un constructeur, 
bref qu’il soit instanciable. 


Nous pouvons donc utiliser cette méthode de cette façon : 


1| Oiseau oiseau = Creer<DUiseau>(); 
2| Avion a380 = Creer<Avion>(); 


Nous appelons la méthode Créer () avec le type générique Oiseau, qui implémente bien 
IVolant et qui est aussi instanciable. Grâce à cela, nous pouvons utiliser l’opérateur 
new pour créer notre type générique, appeler les méthodes de l’interface et renvoyer 
l’instance. Ce qui donne : 


Je déplie mes ailes d’oiseau 
Je bas des ailes 


Je déplie mes ailes mécaniques 
J’allume le moteur 


Si nous tentons d'utiliser la méthode avec un type qui n’implémente pas l’interface 
IVolant, comme : 


1 | Voiture v = Creer <Voiture>(); 


Nous aurons l'erreur de compilation suivante : 


Le type ’MaPremiereApplication.Voiture?” ne peut pas être utilisé 
comme paramètre de type ?’T?’ dans le type ou la méthode géné 
rique ?’MaPremiereApplication.Program.Creer<T>()?. Il n’y a 
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pas de conversion de référence implicite de ? 


MaPremiereApplication.Voiture?” en ?’MaPremiereApplication. 
TVolant 


Globalement, il nous dit que l’objet Voiture n’implémente pas IVolant. 


Oui, mais dans ce cas, plutôt que d'utiliser une méthode générique, pourquoi 
la méthode ne renvoie pas IVolant ? 


C’est une judicieuse remarque, mais elle implique quelques modifications de code. En 
effet, il faudrait indiquer quel type instancier. 


Nous pourrions par exemple faire : 


1] public enum TypeDeVolant 

2! { 

3 Oiseau, 

4 Avion 

51} 

6 

7| public static IVolant Creer (TypeDeVolant type) 
8| { 

9 IVolant volant; 

10 switch (type) 

11 { 

12 case TypeDeVolant .DUiseau: 
13 volant = new Oiseau); 
14 break; 

15 case TypeDeVolant .Avion: 
16 volant = new Avion); 
17 break; 

18 default: 

19 return null; 

20 } 

21 volant.DeplierLesAiles(); 

22 volant.Voler(); 

23 return volant; 

24| } 


Et instancier nos objets de cette façon : 
1| Oiseau oiseau = (Diseau)Creer (TypeDeVolant.0Uiseau); 


2| Avion a380 = (Avion)Creer (TypeDeVolant.Avion); 


Ce qui complique un peu les choses et rajoute des casts dont on pourrait volontiers 
se passer. De plus, si nous créons un nouvel objet qui implémente cette interface, il 
faudrait tout modifier. 


Avouez qu'avec les types génériques, c’est quand même plus propre ! Nous pouvons bien 
sûr avoir des restrictions sur les types génériques d’une classe. 
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Pour le montrer, nous allons créer une classe dont l’objectif est de disposer de types 
valeur qui pourraient ne pas avoir de valeur. Pour les types référence, il suffit d'utiliser 
le mot-clé nul1. Mais pour les types valeur comme les entiers, nous n’avons rien pour 
indiquer que ceux-ci n’ont pas de valeur. 


Par exemple : 


1| public class TypeValeurNull<T> where T : struct 
2| € 

3 private bool aUneValeur; 

4 public bool AUneValeur 

5 { 

6 get Ÿ{ return aUneValeur; } 
7 } 

8 

9 private T valeur; 

10 public T Valeur 

11 { 

12 get 

13 { 

14 if (aUneValeur) 

15 return valeur; 

16 throw new InvalidOperationException(); 
17 

18 set 

19 { 

20 aUneValeur = true; 

o1 valeur = value; 

22 } 

23 } 

24| } 


Ici, nous utilisons une classe possédant un type générique qui sera un type valeur, 
grâce à la condition where T : struct. Cette classe encapsule le type générique pour 
indiquer avec un booléen si le type a une valeur ou pas. Ne faites pas attention à la 
ligne : 


1 | throw new InvalidOperationException(); 


qui permet juste de renvoyer une erreur, nous étudierons les exceptions un peu plus 
loin. Elle pourra s’utiliser ainsi : 


1| TypeValeurNull<int> entier = new TypeValeurNull<int>(); 

2| if (!lentier.AUneValeur) 

31 { 

4 Console.WriteLine("l'entier n'a pas de valeur"); 

5} 

6| entier.Valeur = 5; 

7| if (entier.AUneValeur) 

8| { 

9 Console.WriteLine("Valeur de l'entier : " + entier.Valeur); 
10| } 
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Et si nous souhaitons faire de même pour un autre type valeur, il n’y a rien à coder de 
plus : 


1 | TypeValeurNull<double> valeur = new TypeValeurNull <double >() ; 


C’est quand même super pratique comme classe !! Mais ne rêvons pas, cette idée ne 
vient pas de moi. C’est en fait une fonctionnalité du framework .NET : les types 
nullables. 


Les types nullables 


En fait, la classe que nous avons vue au-dessus existe déjà dans le framework .NET, et 
en mieux! Évidemment. Elle fait exactement ce que j’ai décrit, c’est-à-dire permettre 
à un type valeur d’avoir une valeur nulle. 


Elle est mieux faite dans la mesure où elle tire parti de certaines fonctionnalités du 
framework .NET qui en simplifient l’écriture. Il s’agit de la classe Nullable<>. 


Aussi, nous pourrons créer un entier pouvant être null grâce au code suivant : 


1| Nullable<int> entier = null; 

2| if (lentier.HasValue) 

31 À 

4 Console.WriteLine("l'entier n'a pas de valeur"); 

51 } 

6| entier = 5; 

7| if (entier.HasValue) 

8| À 

9 Console.WriteLine("Valeur de l'entier : " + entier.Value); 
10! } 


Le principe est grosso modo le même sauf que nous pouvons utiliser le mot-clé null 
ou affecter directement la valeur à l’entier en utilisant l’opérateur d'affectation, sans 
passer par la propriété Valeur. Il peut aussi être comparé au mot-clé null ou être 


utilisé avec l'opérateur +, etc. Ceci est possible grâce à certaines fonctionnalités du 
C+# que nous n’avons pas vues et qui sortent de l'étude de ce livre. 


Cette classe est tellement pratique que le compilateur a été optimisé pour simplifier son 
écriture. En effet, utiliser Nullable<> est un peu long pour nous autres informaticiens 
qui sommes des paresseux ! 


Aussi, l'écriture : 


1 | Nullable<int> entier = null; 


peut se simplifier en : 


1 | int? entier = null; 
C’est le point d'interrogation qui remplace la déclaration de la classe Nullable<>. 
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En résumé 


— Avec les génériques, vous pouvez créer des méthodes ou des classes qui sont indépen- 
dantes d’un type. On les appellera des méthodes génériques et des types génériques. 

— On utilise les chevrons <> pour indiquer le type d’une classe ou d’une méthode 
générique. 

— Les interfaces peuvent aussi être génériques, comme l'interface IEnumerable<>. 

— Les types nullables constituent un exemple d’utilisation très pratique des classes 


génériques. 
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TP : types génériques 


Difficulté : 5:53 


h , un peu de pratique histoire de vérifier que nous avons bien compris les génériques. 
C'est un concept assez facile à appréhender mais relativement difficile à mettre en 
œuvre. Quand en ai-je besoin ? Comment ? 


Voici donc un petit exercice qui va vous permettre d'essayer de mettre en œuvre une classe 
générique. 


297 


CHAPITRE 28. TP : TYPES GÉNÉRIQUES 


Instructions pour réaliser la première partie du TP 


Cet exercice va se dérouler en deux parties. Dans la première partie du TP, nous allons 
réaliser une liste chaînée. Il s’agit du grand classique des TP d'informatique en C. Je 
vous en rappelle le principe. 


La liste chaînée permet de naviguer d’élément en élément. Quand nous sommes sur 
le premier élément, le suivant est accessible par sa propriété Suivant. Lorsque nous 
accédons au suivant, l'élément précédent est accessible par la propriété Precedent et 
le suivant toujours accessible par la propriété Suivant. S’il n’y à pas de précédent ou 
pas de suivant, l’élément est null (voir figure 28.1). 


suivant suivant 


précédent précédent 


FIGURE 28.1 — Description de la liste chaînée 


Si on insère un élément à la position 1, les autres se décalent, ainsi qu’illustré à la figure 
282; 


suivant suivant suivant 


CM S 
ER MORT 


précédent précédent précédent 


FIGURE 28.2 — Insertion d’un nouvel élément dans la liste chaînée 


Voilà, il faut donc créer une telle liste chaînée d’éléments. Le but est bien sûr de faire 
en sorte que l’élément soit générique. 


N'hésitez pas à réfléchir un peu avant de vous lancer. Cela pourrait paraître un peu 
simpliste, mais en fait cela occasionne quelques nœuds au cerveau. 


Toujours est-il que je souhaiterais disposer d’une propriété en lecture seule permettant 
d'accéder au premier élément ainsi qu’une autre propriété également en lecture seule 
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permettant d'accéder au dernier élément. Bien sûr, il faut pouvoir naviguer d’élément 
en élément avec des propriétés Precedent et Suivant. 


Il faut évidemment une méthode permettant d’ajouter un élément à la fin de la liste. 
Nous aurons également besoin d’une méthode permettant d’accéder à un élément à 
partir de son indice et enfin d’une méthode permettant d’insérer un élément à un 
indice, décalant tous les suivants. 


Voilà pour la création de la classe. 


Ensuite, notre programme instanciera notre liste chaînée pour lui ajouter les entiers 5, 
10 et 4. Puis nous afficherons les valeurs de cette liste en nous basant sur la première 
propriété et en naviguant d’élément en élément. 


Nous afficherons ensuite les différents éléments en utilisant la méthode d’accès à un 
élément par son indice. 


Enfin, nous insérerons la valeur 99 à la première position (position 0), puis la valeur 
33 à la deuxième position et enfin la valeur 30 à nouveau à la deuxième position. 


Puis nous afficherons tout ce beau monde. 
Fin de l’énoncé, ouf! 


Pour ceux qui n’ont pas besoin d’aide, les explications sont terminées. Ouvrez vos 
Visual C2 Express (ou vos Visual Studio si vous êtes riches !) et à vos claviers. 


Pour les autres, je vais essayer de vous guider un peu plus en essayant tout de même 
de ne pas trop vous donner d'indications non plus. 


En fait, votre liste chaînée n’est pas vraiment une liste, comme pourrait l’être la List<> 
? ? 
que nous connaissons. Cette liste chaînée possède un point d’entrée qui est le premier 
élément. L’ajout du premier élément est très simple, il suffit de mettre à jour une 
propriété. Pour ajouter l’élément suivant, il faut en fait brancher la propriété Suivant 
du premier élément à l’élément que nous sommes en train d’ajouter. Et inversement, 
la propriété Precedent de l’élément que nous souhaitons ajouter sera mise à jour avec 
le premier élément. 


On se rend compte que l’élément est un peu plus complexe qu’un simple type. Nous 
allons donc avoir une classe générique possédant trois propriétés (Precedent, Suivant 
et Valeur). Et nous aurons également une classe du même type générique possédant 
la propriété Premier et la propriété Dernier et les méthodes d’ajout, d'obtention de 
l’élément et d'insertion. 


Allez, je vous en ai assez dit. À vous de jouer! 


Correction 


Pas si facile hein ? Mais bon, comme vous êtes super entraînés, cela n’a pas dû vous 
poser trop de problèmes. 


Voici la correction que je propose. 


La première chose à faire est de créer la classe générique permettant de stocker un 
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élément : 

1| public class Chainage <T> 

2| € 

3 public Chainage<T> Precedent { get; set; } 
4 public Chainage<T> Suivant { get; set; } 

5 public T Valeur { get; set; } 

6| } 


C’est une classe générique toute simple qui possède une valeur du type générique et 
deux propriétés du même type que l'élément pour obtenir le précédent ou le suivant. 


Peut-être que la plus grande difficulté réside, ici, dans le fait de bien modéliser la classe 
qui permet d’encapsuler l’élément. 


Il faudra ensuite créer la liste générique et ses méthodes : 


1| public class ListeChainee <T> 
2| € 
3[ } 


La liste chaînée possède également un type générique. Nous créons sa propriété Premier : 


1| public class ListeChainee <T> 

2| { 

3 public Chainage<T> Premier { get; private set; } 
al} 


Là, c’est très simple, il s’agit juste d’une propriété en lecture seule stockant le premier 
élément. C’est la méthode Ajouter() qui permettra de mettre à jour cette valeur. 
Notez quand même que nous utilisons le type générique comme type générique de la 
classe encapsulante. 


Par contre, pour la propriété Dernier, c’est un peu plus compliqué. Pour la retrouver, 
nous allons parcourir tous les éléments à partir de la propriété Premier. Ce qui donne : 


1| public class ListeChainee <T> 

21 { 

3 [...Code supprimé pour plus de clarté...] 
4 

5 public Chaïinage<T> Dernier 

6 À 

7 get 

8 { 

9 if (Premier == null) 

10 return ouilL: 

11 Chaiïinage <T> dernier = Premier; 
12 while (dernier.Suivant != null) 
13 { 

14 dernier = dernier.Suivant; 
15 } 

16 return dernier; 

17 } 
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18 } 
19 | } 


On parcourt les éléments en bouclant sur la propriété Suivant, tant que celle-ci n’est 
pas nulle. I] s’agit là d’un parcours assez classique où on utilise une variable temporaire 
qui passe au suivant à chaque itération. 


Nous pouvons à présent créer la méthode Ajouter : 


1| public class ListeChainee <T> 

2| € 

3 [...Code supprimé pour plus de clarté...] 

4 

5 public void Ajouter (T element) 

6 { 

7 if (Premier == null) 

8 + 

9 Premier = new Chainage<T> { Valeur = element }; 

10 } 

11 elsé 

12 { 

13 Chainage <T> dernier = Dernier; 

14 dernier.Suivant = new Chainage<T> { Valeur = 
element , Precedent = dernier }; 

15 } 

16 } 

17 | } 


Cette méthode traite tout d’abord le cas du premier élément. Il s’agit simplement de 
mettre à jour la propriété Premier. De même, grâce au calcul interne de la propriété 
Dernier, il sera facile d'ajouter un nouvel élément en se branchant sur la propriété 
Suivant du dernier élément. 


Notez que vu que nous ne la renseignons pas, la propriété Suivant du nouvel élément 
sera bien à null. 


Pour obtenir un élément à un indice donné, il suffira de reprendre le même principe 
que lors du parcours pour obtenir le dernier élément, sauf qu’il faudra s'arrêter au bon 
moment : 


1| public class ListeChaïinee <T> 

2| € 

3 [...Code supprimé pour plus de clarté...] 
4 

5 public Chainage<T> ObtenirElement (int indice) 
6 { 

7 Chaïinage <T> temp = Premier; 

8 for (int i = 1; i <= indice; i++) 

9 Le 

10 if (temp == null) 

11 return null; 

12 temp = temp.Suivant; 
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13 } 

14 return temp; 
15 } 

16| } 


Ici, plusieurs solutions. J’ai choisi d’utiliser une boucle for. Nous aurions très bien pu 
garder la boucle while comme pour la propriété Dernier. 


Enfin, il ne reste plus qu’à insérer un élément : 


1| public class ListeChaïinee <T> 

2| € 

3 [...Code supprimé pour plus de clarté...] 

4 

5 public void Inserer(T element, int indice) 

6 { 

7 if (indice == 0) 

8 À 

9 Chainage <T> temp = Premier; 

10 Premier = new Chainage<T> { Suivant = temp, Valeur 

= element }; 

11 temp.Precedent = Premier; 

12 } 

13 else 

14 { 

15 Chainage <T> elementAlndice = ObtenirElement (indice) 

16 if (elementAlndice == null) 

17 Ajouter (element) ; 

18 else 

19 À 

20 Chainage <T> precedent = elementAlndice. 
Precedent; 

21 Chaiïinage <T> temp = precedent .Suivant; 

22 precedent.Suivant = new Chainage<T> { Valeur = 
element , Precedent = precedent, Suivant = 
temp }; 

23 } 

24 } 

25 } 

26| } 


Nous traitons, dans un premier temps, le cas où l’on doit insérer en tête. Il suffit de 
mettre à jour la valeur du premier en ayant au préalable décalé ce dernier d’un cran. 
Attention, si Premier est null, nous allons avoir un problème. Dans ce cas, soit nous 
laissons le problème; en effet, peut-on vraiment insérer un élément avant les autres 
s’il n’y en a pas? Soit nous gérons le cas et décidons d’insérer l’élément en tant que 
Premier : 


1| public class ListeChainee <T> 
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3 [...Code supprimé pour plus de clarté...] 

4 

5 public void Inserer(T element, int indice) 

6 { 

7 if (indice == 0) 

8 Le 

9 if (Premier == null) 

10 Premier = new Chainage<T> { Valeur = element }; 

11 élsé 

12 { 

13 Chainage <T> temp = Premier; 

14 Premier = new Chainage<T> { Suivant = temp, 
Valeur = element }; 

15 temp.Precedent = Premier; 

16 } 

17 } 

18 else 

19 % 

20 Chainage <T> elementAlndice = ObtenirElement (indice) 

21 if (elementAlndice == null) 

22 Ajouter (element) ; 

23 else 

24 { 

25 Chainage <T> precedent = elementAlndice. 
Precedent; 

26 Chainage <T> temp = precedent . Suivant; 

27 precedent.Suivant = new Chainage<T> { Valeur = 
element , Precedent = precedent, Suivant = 
temp }; 

28 } 

29 } 

30 } 

31| } 


Pour les autres cas, si nous tentons d’insérer à un indice qui n'existe pas, nous insérons 
à la fin en utilisant la méthode Ajouter() existante. Sinon, on intercale le nouvel 
élément dans la liste en prenant soin de brancher le précédent sur notre nouvel élément 


et de brancher le suivant sur notre nouvel élément. 


Voilà pour notre classe. Il reste à l’utiliser : 


1| static void Main(string[] args) 

21 { 

3 ListeChainee<int> listeChainee = new ListeChainee<int>(); 
4 listeChainee.Ajouter (5); 

5 listeChainee.Ajouter (10); 

6 listeChainee.Ajouter (4); 

F Console.WriteLine(listeChainee.Premier.Valeur); 

8 Console.WriteLine(listeChainee.Premier.Suivant.Valeur); 

9 


Console.WriteLine(listeChainee.Premier.Suivant.Suivant. 
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Valeur); 

10 Console.WriteLine ("*#%2%X2%2%%4242% 242% 1) ; 

11 Console.WriteLine(listeChainee.0btenirElement (0).Valeur) ; 
12 Console.WriteLine(listeChainee.0btenirElement (1).Valeur) ; 
13 Console.WriteLine(listeChainee.0ObtenirElement (2) .Valeur); 
14 Console.WriteLine ("#%2%X2% 2% 2440242424 %040%4 1) ; 

15 listeChainee.Inserer (99, O); 

16 listeChainee.Inserer(33, 2); 

17 listeChainee.Inserer(30, 2); 

18 Console.WriteLine(listeChainee.0ObtenirElement (0) .Valeur); 
19 Console.WriteLine(listeChainee.0ObtenirElement (1) .Valeur); 
20 Console.WriteLine(listeChainee.0btenirElement (2).Valeur) ; 
21 Console.WriteLine(listeChainee.0btenirElement (3) .Valeur) ; 
22 Console.WriteLine(listeChainee.0btenirElement (4) .Valeur) ; 
23 Console.WriteLine(listeChainee.0ObtenirElement (5) .Valeur); 


24| } 


Ce qui affichera donc : 


5 

10 

4 
HO Ke 
5 

10 

4 


X XX KO HK OK K OK KE K 


Instructions pour réaliser la deuxième partie du TP 


Bon, c’est très bien de pouvoir accéder à un élément par son indice. Mais une liste sur 
laquelle on ne peut pas faire un foreach, c’est quand même bien dommage. 


Attaquons désormais la deuxième partie du TP. Toujours dans l’optique de manipuler 
les génériques, nous allons faire en sorte que notre liste chaînée puisse être parcourue 
en utilisant un foreach. 


Nous avons dit plus haut qu'il suffisait d’implémenter l'interface IEnumerable. En 
l’occurrence, nous allons implémenter sa version générique, vu que nous travaillons 
avec une classe générique. 


Voilà le but de ce TP! 


304 


INSTRUCTIONS POUR RÉALISER LA DEUXIÈME PARTIE DU TP 


Si vous vous le sentez, allez-y ! Je pense quand même que vous allez avoir besoin d’être 
un peu guidés car c’est une opération assez particulière. 


Vous l’aurez deviné, il faut que notre liste implémente l’interface IEnumerable<T>. 


Le fait d’implémenter cette interface va vous forcer à implémenter deux méthodes 
GetEnumerator (), la version normale et la version explicite. Sachez dès à présent que 
les deux méthodes feront exactement la même chose. 


Q Mais, qu'est-ce qu'il raconte ? Implémenter une interface explicitement ? On 


n'a jamais vu ça! 


C’est vrai! Allez, je vous en parle après la correction. Pour l'instant, cela ne devrait 
pas vous perturber, car les deux méthodes font exactement la même chose. 


En l'occurrence, elles renverront un Enumerator personnalisé. 


Il va donc falloir créer cet énumérateur qui va s'occuper de la mécanique permettant 
de naviguer dans notre liste. Il s’agit d’une nouvelle classe qui va devoir implémenter 
l’interface IEnumerator<T>, c’est-à-dire : 


1] public class ListeChaineeEnumerator <T> : IEnumerator <T> 
2| € 
3| } 


Cette interface permet d'indiquer que notre énumérateur va respecter le contrat lui 
permettant de fonctionner avec un foreach. Avec cette interface, vous allez devoir 
implémenter : 


— la propriété Current ; 

— la propriété explicite Current (qui sera la même chose que la précédente) ; 
— la méthode MoveNext qui permet de passer à l’élément suivant ; 

— la méthode Reset, qui permet de revenir au début de la liste; 

— la méthode Dispose. 


La méthode Dispose est en fait héritée de l'interface IDisposable dont hérite l’inter- 
face IEnumerator<T>. C’est une interface particulière qui offre l'opportunité de faire 
tout ce qu’il faut pour nettoyer la classe, c’est-à-dire libérer les variables qui en auraient 
besoin. En l’occurrence, ici nous n’aurons rien à faire mais il faut quand même que la 
méthode soit présente. Elle sera donc vide. 


Pour implémenter les autres méthodes, il faut que l’énumérateur connaisse la liste 
qu'il doit énumérer. Il faudra donc que la classe ListeChaineeEnumerator prenne en 
paramètre de son constructeur la liste à énumérer. Dans ce constructeur, on initialise 
la variable membre indice qui contient l’indice courant. La propriété Current renverra 
l’élément à l'indice courant. La méthode MoveNext passe à l’élément suivant et renvoie 
faux s’il n’y a plus d'éléments et vrai sinon. Enfin la méthode Reset repasse l'indice à 
sa valeur initiale. 


À noter que la valeur initiale de l’indice est -1, car la boucle foreach commence par 
appeler la méthode MoveNext qui commence par aller à l’élément suivant, c’est-à-dire 
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à l'élément 0. 
Il ne reste plus qu’à vous dire exactement quoi mettre dans les méthodes GetEnumerator 
de la liste chaînée, car vous ne trouverez peut-être pas du premier coup : 


1| public IEnumerator <T> GetEnumerator () 

21 { 

a return new ListeChaineeEnumerator <T>(this); 
al } 

5 

6| IEnumerator IEnumerable.GetEnumerator() 

7| { 

8 return new ListeChaineeEnumerator <T>(this); 
9! } 


C’est à vous de jouer pour la suite. 


Correction 


Encore moins facile ! Tant qu’on ne l’a pas fait une première fois, implémenter l'interface 
IEnumerable est un peu déroutant. Après, c’est toujours pareil. 


Voici donc ma correction. 


Tout d’abord, la liste chaînée doit implémenter IEnumerable<T>, ce qui donne : 


1| public class ListeChaïinee <T> : IEnumerable <T> 

21 { 

3 [...Code identique au TP précédent...] 

4 

5 public IEnumerator <T> GetEnumerator () 

6 { 

7 return new ListeChaineeEnumerator <T>(this); 
8 } 

9 

10 IEnumerator IEnumerable.GetEnumerator () 

11 { 

12 return new ListeChaineeEnumerator <T>(this); 
13 } 


14 } 


Là, c’est du tout cuit vu que je vous avais donné la solution un peu plus tôt . J'espère 
que vous avez au moins réussi ça! 


Maintenant, il faut donc créer un nouvel énumérateur personnalisé en lui passant notre 
liste chaînée en paramètre. Cet énumérateur doit implémenter l’interface IEnumerator, 
ce qui donne : 


1] public class ListeChaïineeEnumerator <T> : IEnumerator <T> 
2| € 
3[ } 


CORRECTION 


Comme prévu, il faut donc un constructeur qui prend en paramètre la liste chaînée : 


1] public class ListeChaineeEnumerator <T> : IEnumerator <T> 
2| € 

3 private int indice; 

4 private ListeChainee<T> listeChainee; 

5 public ListeChaineeEnumerator (ListeChainee<T> liste) 
6 { 

7 indice = -1; 

8 listeChainee = liste; 

9 } 

10 

11 public void Dispose() 

12 { 

13 } 

11} 


Cette liste sera enregistrée dans une variable membre de la classe. Tant que nous y 
sommes, nous ajoutons un indice privé que nous initialisons à -1, comme déjà expliqué. 


Notez également que la méthode Dispose() est vide. Il reste à implémenter les autres 
méthodes : 


1] public class ListeChaineeEnumerator <T> : IEnumerator <T> 

2| € 

3 private int indice; 

4 private ListeChainee<T> listeChainee; 

5 public ListeChaineeEnumerator (ListeChainee<T> liste) 

6 { 

7 indice = -1; 

8 listeChainee = liste; 

9 } 

10 

11 public void Dispose() 

12 { 

13 x 

14 

15 public bool MoveNext() 

16 { 

17 indice++; 

18 Chainage <T> element = listeChainee.0btenirElement( 
indice); 

19 return element != null; 

20 } 

21 

22 public T Current 

23 { 

24 get 

25 À 

26 Chainage <T> element = listeChainee.0btenirElement( 

indice); 
27 if (element == null) 
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28 return default (T); 
29 return element.Valeur; 
30 HA 

31 F 

32 

33 object IEnumerator.Current 

34 { 

35 get { return Current; } 

36 } 

37 

38 public void Reset () 

39 { 

40 indice = -1; 

41 je 

a|} 


Commençons par la méthode MoveNext (). Elle passe à l’indice suivant et renvoie faux 
ou vrai, selon qu’on arrive au bout de la liste ou pas. N'oubliez pas que c’est la première 
méthode qui sera appelée dans le foreach, donc pour passer à l’élément suivant, on 
incrémente l’indice pour le positionner à l’élément 0. C’est pour cela que l’indice a été 
initialisé à -1. On utilise ensuite la méthode existante de la liste pour obtenir l’élément 
à un indice afin de savoir si notre liste peut continuer à s’énumérer. 


La propriété Current renvoie l’élément à l'indice courant, pour cela on utilise l’indice 
pour accéder à l’élément courant, en utilisant les méthodes de la liste. L'autre propriété 
Current fait la même chose, il suffit de l’appeler. 


Enfin, la méthode Reset permet de réinitialiser l’énumérateur en retournant à l’indice 
initial. 

Finalement, ce n’est pas si compliqué que ça. Mais il faut avouer que la première fois, 
c’est un peu déroutant ! 


À mon sens, c’est un bon exercice pratique. Peut-être que mes explications ont suffi à 
vous guider. Sans doute avez-vous dû regarder un peu la documentation de TEnumerable 
sur internet. Peut-être avez-vous cherché des ressources traitant du même sujet. Dans 
tous les cas, devoir implémenter une interface du framework .NET est une situation 
que vous allez fréquemment devoir rencontrer. Il est bon de s’y entraîner. 


Aller plus loin 


Vous me direz qu’il fallait le deviner, qu’on avait besoin d’une classe indépendante qui 
permettait de gérer l’énumérateur ! 


En fait, ce n’est pas obligatoire. On peut très bien faire en sorte que notre classe 
gère la liste chaînée et son énumérateur. Il suffit de faire en sorte que la liste chaînée 
implémente également IEnumerator<T> et de gérer la logique à l’intérieur de la classe. 


Par contre, ce n’est pas recommandé. D’une manière générale il est bon qu’une classe 
n'ait à s'occuper que d’une seule chose. On appelle ça le principe de responsabilité 
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unique !. Plus une classe fait de choses, plus une modification impacte les autres choses. 
Ici, il est judicieux de garder le découplage des deux classes. 


Il y à quand même un élément que l’on peut améliorer dans le code de la correction. 
En effet, cette liste n’est pas extrêmement optimisée car lorsque nous obtenons un 
élément, nous reparcourons toute la liste depuis le début, notamment dans le cas de 
la gestion de l’énumérateur. Il pourrait être judicieux qu’à chaque foreach, nous ne 
parcourions pas tous les éléments et qu’on évite d'appeler continuellement la méthode 
ObtenirElement (). Une idée ? Cela pourrait se faire en éliminant l’indice et en utilisant 
une variable de type Chainage<T>, par exemple : 


1] public class ListeChaineeEnumerator <T> : IEnumerator <T> 
2| { 

3 private Chainage<T> courant; 

4 private ListeChainee<T> listeChainee; 
5 public ListeChaineeEnumerator (ListeChainee<T> liste) 
6 { 

7 courant = null; 

8 listeChainee = liste; 

9 } 

10 

11 public void Dispose() 

12 { 

13 } 

14 

15 public bool MoveNext () 

16 { 

17 if (courant == null) 

18 courant = listeChainee.Premier; 
19 else 

20 courant = courant.Suivant; 

21 

22 return courant != null; 

23 } 

24 

25 public T Current 

26 { 

27 get 

28 { 

29 if (courant == null) 

30 return default (T); 

31 return courant.Valeur; 

32 } 

33 } 

34 

35 object IEnumerator.Current 

36 { 

37 get { return Current; } 

38 k 


1. En anglais, SRP, pour Single Responsibility Principle. 
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39 

40 public void Reset () 
41 { 

42 courant = null; 
43 F 

ail } 


Ici, c’est la variable courant qui nous permet d’itérer au fur et à mesure de la liste 
chaînée. C’est le même principe que dans la méthode ObtenirElement, sauf qu’on ne 
reparcourt pas toute la liste à chaque fois. 


Ici, cette optimisation est négligeable pour notre utilisation. Elle peut s’avérer intéres- 
sante si la liste grossit énormément. 


Dans tous les cas, ça ne fait pas de mal d’aller plus vite! 


Remarquons avant de terminer qu’il est possible de simplifier encore la classe grâce à 
un mot-clé que nous découvrirons dans la partie suivante : yield. Il permet de créer 
facilement des énumérateurs. Ce qui fait que le code complet de la liste chaînée pourra 
être : 


1| public class ListeChaïinee<T> : IEnumerable <T> 

2| { 

3 public Chainage<T> Premier { get; private set; } 

4 

5 public Chaïinage<T> Dernier 

6 { 

7 get 

8 { 

9 if (Premier == null) 

10 return null; 

11 Chainage <T> dernier = Premier; 

12 while (dernier.Suivant != null) 

13 { 

14 dernier = dernier.Suivant; 

15 } 

16 return dernier; 

17 } 

18 } 

19 

20 public void Ajouter (T element) 

21 { 

22 if (Premier == null) 

23 { 

24 Premier = new Chainage<T> { Valeur = element }; 

25 } 

26 else 

27 { 

28 Chainage <T> dernier = Dernier; 

29 dernier.Suivant = new Chainage<T> { Valeur = 
element , Precedent = dernier }; 

30 } 
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31 
32 
33 
34 
35 
36 
37 
38 
39 
40 
41 
42 
43 
44 
45 
46 
47 
48 
49 
50 
51 
52 
53 
54 


55 
56 
57 
58 
59 
60 


61 
62 
63 
64 
65 


66 
67 


68 
69 
70 
71 
72 
73 
74 
75 


} 
public Chainage<T> ObtenirElement (int indice) 
{ 
Chaïinage <T> temp = Premier; 
for (int i = 1; i <= indice; i++) 
Le 
if (temp == null) 
return null; 
temp = temp.Suivant; 
} 
return temp; 
} 
public void Inserer(T element, int indice) 
{ 
if (indice == 0) 
À 
if (Premier == null) 
Premier = new Chainage<T> { Valeur = element }; 
else 
Le 
Chaiïinage <T> temp = Premier; 
Premier = new Chainage<T> { Suivant = temp, 
Valeur = element }; 
temp.Precedent = Premier; 
} 
} 
else 
Le 
Chainage <T> elementAlndice = ObtenirElement (indice) 
if (elementAlndice == null) 
Ajouter (element) ; 
else 
a 
Chainage <T> precedent = elementAIndice. 
Precedent; 
Chaïinage <T> temp = precedent . Suivant; 
precedent.Suivant = new Chainage<T> { Valeur = 
element , Precedent = precedent, Suivant = 
temp }; 
} 
} 
} 


public IEnumerator <T> GetEnumerator () 
{ 
Chainage <T> courant = Premier; 
while (courant != null) 
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76 { 

77 yield return courant.Valeur; 
78 courant = courant.Suivant; 
79 } 

80 } 

81 

82 IEnumerator IEnumerable.GetEnumerator () 
83 < 

84 return GetEnumerator(); 

85 } 

86| } 


marquons que nous n’avons plu in isteChaineeEnumerator. L’im- 
Remarquons que nous n’avons plus besoin de la classe ListeCh E tor. L’ 
plémentation devient très facile. Nous reviendrons sur ce mot-clé dans la partie suivante. 


Implémenter une interface explicitement 


J’en profite ici pour faire un aparté sur l’implémentation d'interface explicitement. 


J’ai choisi délibérément de ne pas le mettre dans le chapitre des interfaces car c’est un 
cas relativement rare mais qui se produit justement quand on implémente l'interface 
IEnumerable<T>. 


Cela vient du fait que l'interface IEnumerable, non générique, expose une propriété 
Current. De même, l'interface IEnumerable<T>, générique, qui hérite de IEnumerable, 
expose également une propriété Current. 


Il y a donc une ambiguïté car les deux propriétés portent le même nom, mais ne 
renvoient pas la même chose. Ce qui est contraire aux règles que nous avons déjà vues. 
Pour faire la différence, il suffira de préfixer la propriété par le nom de l'interface et de 
ne pas mettre le mot-clé public. 


L’implémentation explicite a également un intérêt dans le code suivant : 


1| public interface ICarnivore 

2| € 

3 void Manger); 

al} 

5 

6| public interface IFrugivore 

7| { 

8 void Manger); 

|} 

10 

11| public class Homme : ICarnivore, IFrugivore 
12| 

13 public void Manger() 

14 { 

15 Console.WriteLine("Je mange"); 
16 } 


IMPLÉMENTER UNE INTERFACE EXPLICITEMENT 


17 | } 

18 

19| class Program 

20 | 

21 static void Main(string[]l args) 
22 { 

23 Homme homme = new Homme); 

24 homme .Manger () ; 

25 ((ICarnivore)homme) .Manger () ; 
26 ((IFrugivore)homme) .Manger () ; 
27 } 

28| } 


Ici, ce code compile car la classe Homme implémente la méthode Manger qui est commune 
aux deux interfaces. Par contre, il n’est pas possible de faire la distinction entre le fait 
de manger en tant qu’homme, en tant que ICarnivore ou en tant que IFrugivore. 


Ce code affichera : 


Je mange 


Je mange 
Je mange 


Si c’est le comportement attendu, tant mieux. Si ce n’est pas le cas, il va falloir implé- 
menter au moins une des interfaces de manière explicite : 


1| public class Homme : ICarnivore, IFrugivore 
2| € 
3 public void Manger() 
4 { 
5 Console.WriteLine("Je mange"); 
6 } 
7 
8 void IFrugivore.Manger() 
9 { 
10 Console.WriteLine("Je mange en tant que IFrugivore"); 
11 } 
12 
13 void ICarnivore.Manger() 
14 { 
15 Console.WriteLine("Je mange en tant que ICarnivore"); 
16 } 
17| } 


Avec ce code, notre exemple affichera : 


mange 
mange en tant que ICarnivore 


mange en tant que IFrugivore 


313 


CHAPITRE 28. TP : TYPES GÉNÉRIQUES 


Si vous vous rappelez, nous avions vu au moment du chapitre sur les interfaces que Vi- 
sual C# Express nous proposait de nous aider dans l’implémentation de l’interface. Par 
le bouton droit, vous aviez également accès au sous-menu implémenter l’interface 
explicitement. Vous pouvez vous en servir dans ce cas précis. 


Je m’arrête là sur l’implémentation d’une interface explicite, même s’il y aurait d’autres 
points à voir. Globalement, en situation réelle, cela ne vous servira jamais. 


Voilà pour ce TP. Nous avons créé une classe générique permettant de gérer les listes 
chaînées. Ceci nous à permis de manipuler ces types ô combien indispensables et de nous 
entraîner à la généricité. Nous en avons même profité pour voir comment faire en sorte 
qu’une classe soit énumérable, en implémentant la version générique de IEnumerable. 


Notez bien sûr que cette classe est fonctionnellement incomplète. Il aurait été judicieux 
de rajouter une méthode permettant de supprimer un élément par exemple. À noter 
qu'une classe qui fait à peu près le même travail existe dans le framework .NET, elle 
s'appelle LinkedList. Vous trouverez sa documentation via le code web suivant : 


Code web : 607852 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code ) 


Documentation LinkedList ) 


Code web : 972348 
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Les méthodes d'extension 


Difficulté : 1 


n général, pour ajouter des fonctionnalités à une classe, nous pourrons soit modifier 
le code source de la classe, soit créer une classe dérivée de notre classe et ajouter ces 
fonctionnalités. 


Dans ce chapitre nous allons voir qu'il existe un autre moyen pour étendre une classe : ceci 
est possible grâce aux méthodes d'extension. Elles sont intéressantes si nous n'avons pas 
la main sur le code source de la classe ou si la classe n'est pas dérivable. 


Partons à la découverte de ces fameuses méthodes... 


LE | 
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Qu'est-ce qu’une méthode d’extension ? 


Comme son nom l'indique, une méthode d’extension permet d’étendre une classe en lui 
rajoutant des méthodes. C’est pratique lorsque nous ne possédons pas le code source 
d’une classe et qu’il s’avère difficile de la modifier. 


D'une manière générale, lorsqu'on souhaite modifier une classe dont on n’a pas le code 
source, on utilise une classe dérivée ; ce qui est impossible avec des objets qui ne sont pas 
dérivables, comme par exemple les structures telles que int ou double ou également 
avec la classe String. En effet, String n’est pas dérivable. Nous verrons plus tard 
comment c’est possible, mais pour l’instant, admettons-le. 


Si vous ne me croyez pas, vous pouvez toujours tenter de compiler le code suivant : 


1] public class StringEvoluee : String 
2| { 
3| } 


Créer une méthode d’extension 


Si par exemple nous souhaitons créer une méthode permettant de crypter une chaîne de 
caractères dans un format que nous seuls comprenons, il serait judicieux de créer une 
méthode dans une classe StringCryptee qui dérive de String. Comme ceci n’est pas 
possible, la seule chose qu’il nous reste, c’est de créer une méthode statique utilitaire 
faisant cet encodage : 


1| class Program 

2| € 

3 static void Main(string[]l args) 

4 { 

5 string chaineNormale = "Bonjour à tous"; 

6 string chaineCryptee = Encodage.Crypte(chaineNormale) ; 

7 Console.WriteLine (chaineCryptee); 

8 chaineNormale = Encodage .Decrypte(chaineCryptee); 

9 Console.WriteLine (chaineNormale); 

10 } 

11 

12| } 

13 

14] public static class Encodage 

15| { 

16 public static string Crypte(string chaine) 

17 { 

18 return Convert.ToBase64String (Encoding.Default.GetBytes 
(chaine) ); 

19 } 

20 

21 public static string Decrypte(string chaine) 

22 { 


UTILISER UNE MÉTHODE D'EXTENSION 


23 return Encoding.Default.GetString (Convert. 
FromBase64String(chaine)); 

24 } 

25| } 


Bon, je vous l’accorde, notre encodage secret ne l’est pas tant que ça! Il s’avère que j’uti- 
lise ici un encodage en base 64, algorithme archiconnu. Mais bon, c’est pour l'exemple! 


Utiliser une méthode d’extension 


Ces méthodes statiques jouent bien leur rôle. Mais il est possible de faire en sorte que 
ces deux méthodes deviennent des méthodes d'extension de la classe String. Il suffit 
d'utiliser le mot-clé this devant le premier paramètre de la méthode afin de créer une 
méthode d'extension : 


1] public static class Encodage 

2| € 

3 public static string Crypte(this string chaine) 

4 { 

5 return Convert.ToBase64String (Encoding.Default.GetBytes 
(chaine) ); 

6 1 

T 

8 public static string Decrypte(this string chaine) 

9 { 

10 return Encoding.Default.GetString (Convert. 
FromBase64String(chaine)); 

11 } 

12| } 


Nous pourrons désormais utiliser ces méthodes comme si elles faisaient partie de la 
classe String : 


string chaineNormale = "Bonjour à tous"; 
string chaineCryptee = chaineNormale.Crypte(); 
Console.WriteLine (chaineCryptee); 
chaineNormale = chaineCryptee.Decrypte(); 
Console.WriteLine(chaineNormale); 


a & © ND 


Pas mal non? De plus, si nous regardons dans la complétion automatique, nous pour- 
rons voir apparaître nos fameuses méthodes (voir figure 29.1). 


Plutôt pratique. Évidemment, en créant une méthode d’extension, nous n’avons pas 
accès aux méthodes privées ou variables membres internes à la classe. La preuve, les 
méthodes d’extension sont des méthodes statiques qui travaillent hors de toute instance 
de classe. 


Ces méthodes doivent donc être statiques et situées à l'intérieur d'une classe 
statique. 
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static void Main(string[] args) 


* string chaineNormale = "Bonjour à tous"; 
string chaineCryptee = chaineNormale.Crypte(); 
Console.WriteLine(chaineCryptee); 
chaineNormale = chaineCryptee.; 
Console.WriteLine(chaineNormal Ca Contains<> : 
} © CopyTo 
Ÿ; Count<> M) 
% Crypte 
% Decrypte | (extension) string string.Decrypte() 
Ÿ; DefaultfEmpty<> 
Ÿ; Distinct<> 


G- Flementate> 


FIGURE 29.1 — Notre méthode d'extension apparaît dans la complétion automatique 


Par contre, il faut faire attention à l’espace de nom où se situent nos méthodes d’ex- 
tension. Si le using correspondant n’est pas inclus, nous ne verrons pas les méthodes 
d'extension. 


Remarquez que les méthodes d’extension fonctionnent aussi avec les interfaces. Plus 
précisément, elles viennent étendre toutes les classes qui implémentent une interface. 
Par exemple, avec deux classes implémentant l'interface IVolant : 


1| public interface IVolant 

21 { 

3 void Voler(); 

al} 

5 

6| public class Oiseau : IVolant 

7| À 

8 public void Voler() 

9 { 

10 Console.WriteLine ("Je vole"); 
11 } 

121 } 

13 

14| public class Avion : IVolant 

15] { 

16 public void Voler() 

17 { 

18 Console.WriteLine ("Je vole"); 
19 } 

20 | } 


Si je crée une méthode d'extension prenant en paramètres un IVolant, préfixé par 
this : 


1| public static class Extentions 

2| € 

3 public static void Atterrir(this IVolant volant) 
4 { 

5 Console.WriteLine ("J'atterris"); 


318 


UTILISER UNE MÉTHODE D'EXTENSION 


6 } 
7|} 


Je peux ensuite accéder à cette méthode pour les objets Avion et Diseau : 


1| Avion a = new Avion); 

2| Oiseau b = new DUiseau); 
3| a.Atterrir(); 

4| b.Atterrir(); 


À noter que le framework .NET se sert de ceci pour proposer un grand nombre de 
méthodes d’extension sur les objets implémentant IEnumerable ; nous les étudierons 
un peu plus tard. 


En résumé 


— Une méthode d'extension permet d'étendre une classe en lui rajoutant des méthodes. 

— Il n’est pas recommandé d'utiliser des méthodes d’extension lorsqu'on dispose déjà 
du code source de la classe ou qu’on peut facilement en créer un type dérivé. 

— On utilise le mot-clé this en premier paramètre d’une classe statique pour étendre 
une classe. 
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Délégués, événements et expressions 
lambdas 


Difficulté : 1 


ans ce chapitre, nous allons aborder les délégués, les événements et les expressions 

lambdas. Les délégués et les événements sont des types du framework .NET que nous 

n'avons pas encore vus. Ils permettent d'adresser des solutions notamment dans le 
cadre d'une programmation par événements, comme c'est le cas lorsque nous réalisons des 
applications nécessitant de réagir à une action faite par un utilisateur. Nous verrons dans 
ce chapitre que les expressions lambdas vont de pair avec les délégués. 
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Les délégués (delegate) 


Les délégués ! en C# ne s'occupent pas de la classe, ni du personnel ! Ils permettent de 
créer des variables spéciales. Ce sont des variables qui « pointent » vers une méthode. 
C’est un peu comme les pointeurs de fonctions en € ou C++, sauf qu'ici on sait 
exactement ce que l’on utilise car le C# est fortement typé. 


Le délégué va nous permettre de définir une signature de méthode et avec lui, nous 
pourrons pointer vers n'importe quelle méthode qui respecte cette signature. 


En général, on utilise un délégué quand on veut passer une méthode en paramètre d’une 
autre méthode. Un petit exemple sera sans doute plus parlant qu’un long discours. 
Ainsi, le code suivant : 


1| public class TrieurDeTableau 

2| { 

3 private delegate void DelegateTri(int[]l tableau); 
al} 


crée un délégué privé à la classe TrieurDeTableau qui permettra de pointer vers des 
méthodes qui ne retournent rien (void) et qui acceptent un tableau d’entiers en para- 
mètre. 

C’est justement le cas des méthodes TriAscendant () et TriDescendant() que nous 
allons ajouter à la classe (ça tombe bien!) : 


1| public class TrieurDeTableau 

2| € 

3 private delegate void DelegateTri(int[] tableau); 
4 

5 private void TriAscendant (int [] tableau) 
6 { 

7 Array.Sort (tableau) ; 

8 } 

9 

10 private void TriDescendant (int [] tableau) 
11 { 

12 Array.Sort (tableau); 

13 Array.Reverse (tableau); 

14 } 

15| } 


Vous aurez compris que la méthode TriAscendant utilise la méthode Array.Sort pour 
trier un tableau par ordre croissant. Inversement, la méthode TriDescendant() trie le 
tableau par ordre décroissant en triant par ordre croissant et en inversant ensuite le 
tableau. 


Il ne reste plus qu’à créer une méthode dans la classe permettant d'utiliser le tri as- 
cendant et le tri descendant, grâce à notre délégué : 


1 | public class TrieurDeTableau 


1. En anglais, delegate. 
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2! { 

3 [...Code supprimé pour plus de clarté...] 
4 

5 public void DemoTri(int[] tableau) 
6 { 

7 DelegateTri tri = TriAscendant; 
8 tri(tableau); 

9 foreach (int i in tableau) 

10 { 

11 Console.WriteLine(i); 

12 } 

13 

14 Console.WriteLine(); 

15 tri = TriDescendant; 

16 tri(tableau); 

17 foreach (int i in tableau) 

18 { 

19 Console.WriteLine(i); 

20 } 

21 } 

22 | } 


Nous voyons ici que dans la méthode DemoTri nous commençons par déclarer une 
variable du type du délégué DelegateTri, qui est le délégué que nous avons créé. Puis 
nous faisons pointer cette variable vers la méthode TriAscendant(). 


Nul besoin ici d'utiliser les parenthèses, mais juste le nom de la méthode. II 
s'agit juste d'une affectation. 


Nous invoquons ensuite la méthode TriAscendant () à travers la variable qui va per- 
mettre de trier le tableau par ordre croissant avant d’afficher son contenu. Cette fois-ci, 
il faut bien sûr utiliser les parenthèses car nous invoquons la méthode. Puis nous faisons 
pointer la variable vers la méthode TriDescendant() qui va nous permettre de faire 
la même chose mais avec un tri décroissant. 


Nous pouvons appeler cette classe de cette façon : 


1| static void Main(string[] args) 

2! { 

3 int [] tableau = new int[] { 4, 1, 6, 10, 8, 5 }; 
4 new TrieurDeTableau().DemoTri(tableau); 

51 } 


Notre code affichera au final les entiers triés par ordre croissant, puis les mêmes entiers 
triés par ordre décroissant. 


Ok, mais pourquoi utiliser ce délégué ? On pourrait très bien appeler d'abord 


la méthode de tri ascendant et ensuite la méthode de tri descendant. Comme 
on l'a toujours fait! 
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Eh bien, l'intérêt ici est que le délégué est très souple et va permettre de réorganiser le 
code (on parle également de refactoriser du code). Ainsi, en rajoutant la méthode 
suivante dans la classe : 


1| private void TrierEtAfficher(int[] tableau, DelegateTri 
methodeDeTri) 


2! À 

3 methodeDeTri(tableau) ; 

4 foreach (int i in tableau) 
5 { 

6 Console.WriteLine(i); 
7 } 

8| } 


nous pourrons grandement simplifier la méthode DemoTri : 


1| public void DemoTri(int[] tableau) 

2| { 

3 TrierEtAfficher (tableau, TriAscendant); 
4 Console.WriteLine(); 

5 TrierEtAfficher (tableau, TriDescendant); 
6 


ui produir même résu ue précédemment. Qu’avons-nous fait ici ! 
Ce oduira le même résultat que précéde ent ’avons-nous fait ici ? 


Nous avons utilisé le délégué comme paramètre d’une méthode. Ce délégué est ensuite 
utilisé pour invoquer une méthode que nous aurons passée en paramètres. C’est ce que 
nous faisons en disant d'utiliser la méthode TrierEtAfficher une première fois avec 
la méthode TriAscendant() et une deuxième fois avec la méthode TriDescendant(). 


Plutôt pas mal non? Il est même possible de définir la méthode qui sera utilisée à 
l’intérieur de TrierEtAfficher() sans avoir à l'écrire complètement dans le corps de 
la classe. 


Cela peut être utile si la méthode n’est vouée à être utilisée que dans cette unique 
situation et qu’elle n’est jamais appelée à un autre endroit. Par exemple, plutôt que de 
définir complètement la méthode TriAscendant(), je pourrais la définir directement 
au moment de l’appel de la méthode : 


1| public class TrieurDeTableau 

2| { 

3 private delegate void DelegateTri(int[]l tableau); 

4 

5 private void TrierEtAfficher(int[] tableau, DelegateTri 
methodeDeTri) 

6 { 

6 methodeDeTri(tableau) ; 

8 foreach (int i in tableau) 

9 { 

10 Console.WriteLine(i); 

11 } 

12 } 


DIFFUSION MULTIPLE, LE MULTICAST 


14 public void DemoTri(int[] tableau) 

15 { 

16 TrierEtAfficher (tableau, delegate(int [] leTableau) 
17 Le 

18 Array.Sort(leTableau); 

19 }); 


21 Console.WriteLine(); 


23 TrierEtAfficher (tableau, delegate(int [] leTableau) 
24 { 

25 Array.Sort (tableau); 

26 Array.Reverse (tableau) ; 


27 YF); 


Ainsi, je n'aurai plus besoin des méthodes TriAscendant () et TriDescendant(). 


Le fait de définir la méthode directement au niveau du paramètre d’appel est ce qu’on 
appelle « utiliser une méthode anonyme ». Anonyme car la méthode n’a pas de 
nom. Elle n’a de vie qu’à cet endroit-là. 


La syntaxe est un peu particulière, mais au lieu d'utiliser une variable de type delegate 
qui pointe vers une méthode, c’est comme si on écrivait directement la méthode. 


On utilise le mot-clé delegate suivi de la déclaration du paramètre. Évidemment, le 
délégué anonyme doit respecter la signature de DelegateTri que nous avons défini 
plus haut. Enfin, nous faisons suivre avec un bloc de code qui correspond au corps de 
la méthode anonyme. 


À noter que le fait d'utiliser le mot-clé delegate revient en fait à créer 


une classe qui dérive de System.Delegate et qui implémente la logique de 
base d'un délégué. Le C# nous masque tout ceci afin d'être le plus efficace 


possible. 


Diffusion multiple, le multicast 


Il faut également savoir que le délégué peut être multicast, cela veut dire qu’il peut 
pointer vers plusieurs méthodes. Améliorons le premier exemple : 


1| public class TrieurDeTableau 

2| € 

3 private delegate void DelegateTri(int[]l tableau); 
4 

5 private void TriAscendant (int [] tableau) 
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6 { 

7 Array.Sort (tableau); 

8 foreach (int i in tableau) 

9 { 

10 Console.WriteLine(i); 

11 } 

12 Console.WriteLine(); 

13 } 

14 

15 private void TriDescendant (int [] tableau) 
16 { 

17 Array.Sort (tableau); 

18 Array.Reverse (tableau); 

19 foreach (int i in tableau) 

20 { 

21 Console.WriteLine(i); 

22 } 

23 } 

24 

25 public void DemoTri(int[] tableau) 
26 { 

27 DelegateTri tri = TriAscendant; 
28 tri += TriDescendant; 

29 tri(tableau); 

30 + 

31| } 


Ici, j'utilise Console .WriteLine directement dans chaque méthode de tri afin de bien 
voir le résultat du tri du tableau. L'important est de voir que dans la méthode DemoTri, 
je commence par créer un délégué que je fais pointer vers la méthode TriAscendant. 
Puis j'ajoute à ce délégué, avec l'opérateur +—, une nouvelle méthode, à savoir 
TriDescendant. Désormais, le fait d’invoquer le délégué va invoquer les deux mé- 
thodes ; ce qui produira en sortie : 


Ce détail prend toute son importance avec les événements que nous verrons plus loin. 
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À noter que le résultat de ce code est évidemment identique en utilisant les méthodes 
anonymes : 


1| public void DemoTri(int[] tableau) 
2| { 

3 DelegateTri tri = delegate(int[] leTableau) 
4 { 

5 Array.Sort(leTableau); 

6 foreach (int i in tableau) 
7 { 

8 Console.WriteLine(i); 
9 } 

10 Console.WriteLine(); 

11 }; 

12 tri += delegate(int[] leTableau) 
13 À 

14 Array.Sort (tableau); 

15 Array.Reverse (tableau); 

16 foreach (int i in tableau) 
17 + 

18 Console.WriteLine(i); 
19 } 

20 }; 

21 tri(tableau); 

22| } 


Il faut quand même remarquer que l'ordre dans lequel sont appelées les mé- 
thodes n'est pas garanti et ne dépend pas forcément de l'ordre dans lequel 
nous les avons ajoutées au délégué. 


Les délégués génériques Action et Func 


C'est très bien tout ça, mais cela veut dire qu'à chaque fois que je vais avoir 
besoin d'utiliser un délégué, je vais devoir créer un nouveau type en utilisant 
le mot-clé delegate ? 


Pas forcément, c’est là qu’interviennent les délégués génériques Action et Func. Action 
est un délégué qui permet de pointer vers une méthode qui ne renvoie rien et qui peut 
accepter jusqu’à 16 types différents. Cela veut dire que le code précédent peut être 
remplacé par : 


1| public class TrieurDeTableau 


2| À 

3 private void TrierEtAfficher(int[] tableau, Action<int[]l> 
methodeDeTri) 

4 { 
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5 methodeDeTri(tableau) ; 

6 foreach (int i in tableau) 

7 { 

8 Console.WriteLine(i); 

9 } 

10 } 

11 

12 public void DemoTri(int[] tableau) 
13 { 

14 TrierEtAfficher (tableau, delegate(int[] leTableau) 
15 { 

16 Array.Sort(leTableau) ; 

17 }) ; 

18 

19 Console.WriteLine(); 

20 

21 TrierEtAfficher (tableau, delegate(int[] leTableau) 
22 { 

23 Array.Sort (tableau) ; 

24 Array.Reverse (tableau); 

25 }) ; 

26 } 

27| } 


Notez que la différence se situe au niveau du paramètre de la méthode TrierEtAfficher 
qui prend un Action<int{[]>. En fait, cela est équivalent à créer un délégué qui ne ren- 
voie rien et qui prend un tableau d’entiers en paramètre. Si notre méthode avait deux 
paramètres, il aurait suffi d'utiliser la forme de Action avec plusieurs paramètres géné- 
riques, par exemple Action<int[]l, string> pour avoir une méthode qui ne renvoie 
rien et qui prend un tableau d’entiers et une chaîne de caractères en paramètres. 


Lorsque la méthode renvoie quelque chose, on peut utiliser le délégué Func<T>, sachant 
qu'ici, c’est le dernier paramètre générique qui sera du type de retour du délégué. Par 
exemple : 


1| public class Operations 

2| € 

3 public void DemoUperations() 

4 { 

5 double division = Calcul(delegate(int a, int b) 
6 { 

7 return (double)a / (double)b; 

8 F5 4 Hi, 

9 

10 double puissance = Calcul(delegate(int a, int b) 
11 { 

12 return Math.Pow((double)a, (double)b); 

13 }, 4, 5); 

14 

15 Console.WriteLine("Division : " + division); 

16 Console.WriteLine("Puissance : " + puissance); 
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17 } 

18 

19 private double Calcul(Func<int, int, double> 
methodeDeCalcul, int a, int b) 

20 { 

21 return methodeDeCalcul(a, bj); 

29 } 

23| } 


Ici, dans la méthode Calcul, on utilise le délégué Func pour indiquer que la méthode 
prend deux entiers en paramètres et renvoie un double. Si nous utilisons cette classe 
avec le code suivant : 


1| class Program 

2| € 

3 static void Main(string[l]l args) 

4 { 

5 new DOperations ().DemoUperations (); 
6 } 

7|} 


nous aurons : 


Division. 105 


Puissance : 1024 


Les expressions lambdas 


Non, il ne s’agit pas d’une expression qui danse la lambada, mais d’une façon simplifiée 
d'écrire les délégués que nous avons vus au-dessus ! 


Ainsi, le code suivant : 


1| DelegateTri tri = delegate(int[] leTableau) 
2| € 

3 Array.Sort(leTableau) ; 

al }; 


peut également s’écrire de cette façon : 


1| DelegateTri tri = (leTableau) => 
2| { 

3 Array.Sort(leTableau) ; 

al}; 


Cette syntaxe est particulière. La variable leTableau permet de spécifier le paramètre 
d'entrée de l'expression lambda. Ce paramètre est écrit entre parenthèses. Ici, pas 
besoin d’indiquer son type vu qu’il est connu par la signature associée au délégué. On 
utilise ensuite la flèche => pour définir l’expression lambda qui sera utilisée. Elle s’écrit 
de la même façon qu’une méthode, dans un bloc de code. 
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L'expression lambda (1eTableau) => se lit : & leTableau conduit à ». 


Dans le corps de la méthode, nous voyons que nous utilisons la variable leTableau de 
la même façon que précédemment. 


Dans ce cas précis, il est encore possible de raccourcir l'écriture car la méthode ne 
contient qu'une seule ligne; on pourra alors l'écrire de cette façon : 


1 | TrierEtAfficher (tableau, leTableau => Array.Sort (leTableau)); 


thèses. 


Q S'il y a un seul paramètre à l'expression lambda, on peut omettre les paren- 


Quand il y a deux paramètres, on les sépare par une virgule. À noter qu’on n’indique 
nulle part le type de retour, s’il y en à un. Notre expression lambda remplaçant le 
calcul de la division peut donc s’écrire ainsi : 


1| double division = Calcul((a, b) => 
NE 

3 return (double)a / (double)b; 
af }, 4, 5); 


Lorsque l’instruction possède une unique ligne, on peut encore en simplifier l’écriture ; 
ce qui donne : 


1| double division = Calcul((a, b) => (double)a / (double)b, 4, 5) 


, 


Q Pourquoi tout ce blabla sur les delegate et les expressions lambdas ? 


Pour deux raisons : 


— Parce que les délégués sont la base des événements. 
— À cause des méthodes d'extension LINQ. 


Nous parlerons dans la partie suivante des méthodes d’extension LINQ. Quant aux 
événements, explorons-les dès maintenant ! 


Les événements 


Les événements sont un mécanisme du C# permettant à une classe d’être notifiée d’un 
changement. Par exemple, on peut vouloir s’abonner à un changement de prix d’une 
voiture. La base des événements est le délégué. On pourra stocker dans un événement 
un ou plusieurs délégués qui pointent vers des méthodes respectant la signature de 
l'événement. 


Un événement est défini grâce au mot-clé event. Prenons cet exemple : 
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public class Voiture 


Æ 


public delegate void DelegateDeChangementDePrix(decimal 


nouveauPrix); 


public event DelegateDeChangementDePrix ChangementDePrix; 
public decimal Prix { get; set; } 


public void PromoSurLePrix() 


A 


Prix = Prix / 2; 
if (ChangementDePrix != null) 
ChangementDePrix(Prix); 


Dans la classe Voiture, nous définissons un délégué qui ne retourne rien et qui prend 
en paramètre un décimal. Nous définissons ensuite un événement basé sur ce délégué, 
avec, comme nous l’avons vu, l’utilisation du mot-clé event. Enfin, dans la méthode de 
promotion, après un changement de prix (division par 2), nous notifions les éventuels 


) 


objets qui se seraient abonnés à cet événement en invoquant l'événement et en lui 
fournissant en paramètre le nouveau prix. 


À noter que nous testons d’abord s’il y a un abonné à l'événement (en testant s’il est 
différent de null) avant de le lever. 


Pour s’abonner à cet événement, il suffit d’utiliser le code suivant : 


class Program 


:S 


static void Main(string[]l args) 


{ 


} 


new DemoEvenement () .Demo(); 


public class DemoEvenement 


{ 


public void Demo) 


€ 


} 


Voiture voiture = new Voiture { Prix = 10000 +}; 
Voiture.DelegateDeChangementDePrix 
delegateChangementDePrix = voiture _ChangementDePrix; 


voiture.ChangementDePrix += delegateChangementDePrix; 


voiture.PromoSurLePrix(); 


private void voiture _ChangementDePrix(decimal nouveauPrix) 


{ 


Console.WriteLine("Le nouveau prix est de : " + 
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nouveauPrix); 


Nous créons une voiture, et nous créons un délégué du même type que l'événement. 
Nous le faisons pointer vers une méthode qui respecte la signature du délégué. Ainsi, à 
chaque changement de prix, la méthode voiture_ChangementDePrix va être appelée 
et le paramètre nouveauPrix possédera le nouveau prix qui vient d’être calculé. 


Appelons la promotion en invoquant la méthode ChangementDePrix(). Nous pouvons 
nous rendre compte que l’application nous affiche le nouveau prix qui est l’ancien divisé 
par 2. 


Lorsque nous commençons à écrire le code qui va permettre de nous abonner à l’évé- 
nement, la complétion automatique nous propose facilement de créer une méthode qui 
respecte la signature de l'événement. Il suffit de saisir l’événement, d’ajouter ensuite 
un += et d'appuyer sur la touche pour que Visual C# Express nous propose de 
tout insérer automatiquement (voir la figure 30.1). 


aa 
#3 MaPremiereApplication.DemoEvenement | % DemoQ 


{ 


class Program 
static void Main(string[] args) 
new DemoEvenement().Demo(); 


} 
} 


B public class Demoëvenement 
public void Demo() 


Voiture voiture = new Voiture { Prix = 10000 }; 
voiture.ChangementDePrix +=| 


voiture. PromosurLeprix(); | new Voiture.DelegateDeChangementDePrix{voiture ChangementDePrix): (Appuyez sur TABULATION pour insérer) 


FIGURE 30.1 - Complétion automatique d'événement 


Ce qui génère le code suivant : 


1| voiture.ChangementDePrix += new Voiture. 
DelegateDeChangementDePrix(voiture_ChangementDePrix); 


ainsi que la méthode : 


1 |vord voiture_ChangementDePrix(decimal nouveauPrix) 
21 { 

3 throw new NotImplementedException(); 

al } 


On peut aisément simplifier l'abonnement avec : 


1 | voiture.ChangementDePrix += voiture_ChangementDePrix; 


comme on l’a déjà vu. Notez que vous pouvez également rajouter la visibilité private 
sur la méthode générée afin que cela soit plus explicite. 


1 | private void voiture_ChangementDePrix(decimal nouveauPrix) 
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2| { 
3[ } 


L'utilisation du += permet d'ajouter un nouveau délégué à l’événement. Il sera éven- 
tuellement possible d'ajouter un autre délégué avec le même opérateur; ainsi deux 
méthodes seront désormais notifiées en cas de changement de prix. 


Inversement, il est possible de se désabonner d’un événement en utilisant l’opérateur 


Les événements sont largement utilisés dans les applications ayant recours au C#, 
comme les applications clients lourds développées avec WPF par exemple. Ce sont des 
applications comme un traitement de texte ou un navigateur internet. Par exemple, 
lorsque l’on clique sur un bouton, un événement est levé. 


Ces événements utilisent en général une construction à base du délégué EventHandler 
ou sa version générique EventHandler<>. Ce délégué accepte deux paramètres. Le 
premier de type object qui représente la source de l’événement, c’est-à-dire l’objet qui 
a levé l'événement. Le second est une classe qui dérive de la classe de base EventArgs. 


Réécrivons notre exemple avec ce nouveau handler. Nous avons donc besoin en premier 
lieu d’une classe qui dérive de la classe EventArgs : 


1] public class ChangementDePrixEventArgs : EventArgs 
2| { 

3 public decimal Prix { get; set; } 

al} 


Plus besoin de déclaration de délégué, nous utilisons directement EventHandler dans 
notre classe Voiture : 


1| public class Voiture 


2| € 

3 public event EventHandler <ChangementDePrixEventArgs > 
ChangementDePrix; 

4 public decimal Prix { get; set; } 

5 

6 public void PromoSurLePrix() 

Le { 

8 Prix = Prix / 2; 

9 if (ChangementDePrix != null) 

10 ChangementDePrix(this, new 

ChangementDePrixEventArgs { Prix = Prix }); 
11 } 
12| } 


Et notre démo devient : 


1] class Program 

2| € 

3 static void Main(string[l]l args) 
4 { 

5 new DemoEvenement () .Demo(); 
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6 } 

7|} 

8 

9| public class DemoEvenement 

10| 

11 public void Demo) 

12 { 

13 Voiture voiture = new Voiture { Prix = 10000 }; 

14 

15 voiture.ChangementDePrix += voiture _ChangementDePrix; 

16 

17 voiture.PromoSurLePrix(); 

18 } 

19 

20 private void voiture _ChangementDePrix(object sender, 
ChangementDePrixEventArgs e) 

21 { 

22 Console.WriteLine ("Le nouveau prix est de : " + e.Prix) 

23 } 

24| } 


Remarquez que la méthode voiture_ChangementDePrix prend désormais deux para- 
mètres. Le premier représente l’objet Voiture ; si nous en avions besoin, nous pourrions 
l’utiliser avec un cast adéquat. Le second représente l’objet contenant le prix de la voi- 
ture en promotion. 


À noter que, de manière générale, nous allons surtout utiliser les événements définis 
? ? 
par le framework .NET. Il sera donc assez rare d’avoir à en définir un soi-même. 


En résumé 


— Les délégués permettent de créer des variables pointant vers des méthodes. 

— Les délégués sont à la base des événements. 

On utilise les expressions lambdas pour simplifier l'écriture des délégués. 

— Les événements sont un mécanisme du C# permettant à une classe d’être notifiée 
d’un changement. 
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 } EE 


Gérer les erreurs : les exceptions 


Difficulté : 


ous avons parlé rapidement des erreurs dans nos applications C# en disant qu'il 

s'agissait d'exceptions. C'est le moment d'en savoir un peu plus et surtout d'apprendre 

à les gérer! Dans ce chapitre, nous allons apprendre comment créer et intercepter 
une exception. 
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Intercepter une exception 


Rappelez-vous de ce code : 


1| string chaine = "dix"; 
2| int valeur = Convert.Tolnt32(chaine); 


Si nous l’exécutons, nous aurons un message d’erreur : 


Exception non gérée : System.FormatException: Le format de la 
chaîne d’entrée est incorrect. 

à System.Number.StringToNumber (String str, NumberStyles options, 
NumberBuffer 


number , NumberFormatIlnfo info, Boolean parseDecimal) 
System.Number.Parselnt32(String s, NumberStyles style, 


NumberFormatInfo info) 
System.Convert.ToInt32(String value) 
MaPremiereApplication.Program.Main(String[] args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
14 


L'application nous affiche un message d’erreur et plante lamentablement, produisant 
par la même occasion un rapport d’erreur. Ce qui se passe en fait, c’est que lors de 
la conversion, si le framework .NET n'arrive pas à convertir correctement la chaîne de 
caractères en entier, il lève une exception. Cela veut dire qu’il informe le programme 
qu'il rencontre un cas limite qui nécessite d’être géré. Si ce cas limite n’est pas géré, 
alors l’application plante et c’est le CLR qui intercepte l'erreur et fait produire un 
rapport au système d'exploitation. 


Pourquoi une exception et pas un message d'erreur ? 


L'intérêt des exceptions est qu’elles sont typées. C’en est fini des codes d’erreurs incom- 
préhensibles. C’est le type de l’exception, c’est-à-dire sa classe, qui va nous permettre 
d'identifier le problème. 


Pour éviter le plantage de l'application, nous devons gérer ces cas limites et intercepter 
les exceptions. Pour ce faire, il faut encadrer les instructions pouvant atteindre des cas 
limites avec le bloc d'instruction try...catch, par exemple : 


1| try 

2| € 

3 string chaïine = "dix"; 

4 int valeur = Convert.Tolnt32(chaine); 

5 Console.WriteLine("Ce code ne sera jamais affiché"); 
6| } 

7| catch (Exception) 

8| { 
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9 Console.WriteLine ("Une erreur s'est produite dans la 
tentative de conversion"); 
10! } 


Si nous exécutons ce bout de code, l’application ne plantera plus et affichera qu’une 
erreur s’est produite... 


Nous avons « attrapé » l'exception levée par la méthode de conversion grâce au mot- 
clé catch. Cette construction nous permet de surveiller l'exécution d’un bout de code, 
situé dans le bloc try et, s’il y a une erreur, alors nous interrompons son exécution 
pour traiter l’erreur en allant dans le bloc catch. La suite du code dans le try, à savoir 
l'affichage de la ligne avec Console.WriteLine, ne sera jamais exécuté, car lorsque la 
conversion échoue, il saute directement au bloc catch. Inversement, il est possible de 
ne jamais passer dans le bloc catch si les instructions ne provoquent pas d’erreur : 


1| try 

2| { 

3 string chaine = "10"; 

4 int valeur = Convert.Tolnt32(chaine); 

5 Console.WriteLine ("Conversion OK"); 

6| } 

7| catch (Exception) 

8| { 

9 Console.WriteLine("Nous ne passons jamais ici ..."); 
10! } 


Ainsi le code ci-dessus affichera uniquement que la conversion est bonne et, en toute 
logique, il ne passera pas dans le bloc de traitement d'erreur, car il n’y en a pas eu. Il 
est possible d'obtenir des informations sur l’exception en utilisant un paramètre dans 
le bloc catch. Ce paramètre est une variable du type Exception, qui dans l’exemple 
suivant s'appelle ex : 


1| try 

2| € 

3 string chaine = "dix"; 

4 int valeur = Convert.Tolnt32(chaine); 

5| } 

6| catch (Exception ex) 

al 

8 Console.WriteLine("Il y a un eu une erreur, plus d' 
informations ci-dessous :"); 

9 Console.WriteLine); 

10 Console.WriteLine ("Message d'erreur : " + ex.Message); 

11 Console.WriteLine); 

12 Console.WriteLine("Pile d'appel : " + ex.StackTrace); 

13 Console.WriteLine(); 

14 Console.WriteLine ("Type de l'exception : " + ex.GetType()); 

15| } 


Ici, nous affichons le message d’erreur, la pile d’appel et le type de l'exception, ce qui 
donne le message suivant : 
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I1 y a un eu une erreur, plus d’informations ci-dessous 


Message d’erreur : Le format de la chaîne d’entrée est incorrect 


Pile d’appel : à System.Number.StringToNumber (String str, 
NumberStyles options, NumberBuffer& number, NumberFormatInfo 
info, Boolean parseDecimal) 


System.Number.Parselnt32(String s, NumberStyles style, 
NumberFormatInfo info) 

System.Convert.Tolnt32(String value) 

MaPremiereApplication.Program.Main(String[] args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
16 


Type de l’exception : System.FormatException 


D'une manière générale, la méthode ToString() de l’exception fournit des informations 
suffisantes pour identifier l'erreur : 


LL Cry 

2| { 

3 string chaine = "dix"; 

4 int valeur = Convert.Tolnt32(chaine); 

5|} 

6| catch (Exception ex) 

7| t 

8 Console.WriteLine("Il y a un eu une erreur : " + ex. 
ToString()); 

o[} 


Ce code affichera : 


I1 y a un eu une erreur : System.FormatException: Le format de 
la chaîne d’entrée est incorrect. 
System.Number.StringToNumber (String str, NumberStyles options, 
NumberBuffer 
number , NumberFormatIlnfo info, Boolean parseDecimal) 
System.Number.Parselnt32(String s, NumberStyles style, 


NumberFormatInfo info) 
System.Convert.Tolnt32(String value) 
MaPremiereApplication.Program.Main(String[] args) dans C:\ 
Users\Nico\Documents\Visual Studio 2010\Projects\C#\ 
MaPremiereApplication\MaPremiereApplication\Program.cs:ligne 
16 


Les exceptions peuvent être de beaucoup de formes. Ici nous remarquons que l’exception 
est de type System.FormatException. Cette exception est utilisée en général lorsque 
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le format d’un paramètre ne correspond pas à ce qui est attendu. En l’occurrence ici, 
nous attendons un paramètre de type chaîne de caractères qui représente un entier. 
C’est une exception spécifique qui est dédiée à un type d’erreur précis. 


Il faut savoir que comme beaucoup d’autres objets du framework .NET, les exceptions 
spécifiques dérivent d’une classe de base, à savoir la classe Exception, dont vous pouvez 
trouver la documentation via le code web suivant : 


Exceptions ) 


Code web : 368326 


Il existe une hiérarchie entre les exceptions. Globalement, deux grandes familles d’ex- 
ceptions existent : ApplicationException et SystemException. Elles dérivent toutes 
les deux de la classe de base Exception. La première est utilisée lorsque des erreurs 
récupérables sur des applications apparaissent ; la seconde est utilisée pour toutes les 
exceptions générées par le framework .NET. Par exemple, l'exception que nous avons 
vue, FormatException dérive directement de SystemException, qui dérive elle-même 
de la classe Exception. 


Le framework .NET dispose de beaucoup d’exceptions correspondant à beaucoup de 
situations. Notons encore au passage une autre exception bien connue des développeurs 
qui est la NullReferenceException. Elle se produit lorsqu'on essaie d’accéder à un 
objet qui vaut null. Par exemple : 


1] Voiture voiture = null; 
2| voiture.Vitesse 10; 


Vous aurez remarqué que dans la construction suivante : 


1| try 

2| € 

3 string chaine = "dix"; 

4 int valeur = Convert.Tolnt32(chaine); 

5| } 

6| catch (Exception ex) 

7| T 

8 Console.WriteLine("Il y a un eu une erreur : " + ex. 
ToString()); 

9} 


nous voyons que le bloc catch prend en paramètre la classe de base Exception. 
Cela veut dire que nous souhaitons intercepter toutes les exceptions qui dérivent de 
Exception: c’est-à-dire en fait toutes les exceptions, car toute exception dérive forcé- 
ment de la classe Exception. C’est utile lorsque nous voulons attraper toutes les ex- 
ceptions. Mais savons-nous forcément quoi faire dans le cas de toutes les erreurs ? Il est 
possible d’être plus précis afin de n’attraper qu’un seul type d'exception. Il suffit de pré- 
ciser le type de l’exception attendue comme paramètre du catch. Par exemple le code 
suivant nous permet d’intercepter toutes les exceptions du type FormatException : 


1| try 
2| { 
3 string chaine = "dix"; 
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4 int valeur = Convert.Tolnt32(chaine); 
5| } 

6| catch (FormatException ex) 

7|{ 

8 Console.WriteLine (ex); 

9! } 


Cela veut par contre dire que si nous avons une autre exception à ce moment-là, du style 
NullReferenceException, elle ne sera pas attrapée ; ce qui fait que le code suivant va 
planter : 


1| try 

21 { 

à Voiture v = null; 

4 v.Vitesse = 10; 

51 } 

6| catch (FormatException ex) 

7| { 

8 Console.WriteLine ("Erreur de format : " + ex); 
9! } 


En effet, nous demandons uniquement la surveillance de l’exception FormatException. 
Ainsi, l'exception NullReferenceException ne sera pas attrapée. 


Intercepter plusieurs exceptions 


Pour attraper les deux exceptions, il est possible d’enchaîner les blocs catch avec des 
paramètres différents : 


L | CET 

2| { 

3 // code provoquant une exception 

al } 

5| catch (FormatException ex) 

6| { 

7 Console.WriteLine("Erreur de format : " + ex); 
8| } 

9| catch (NullReferenceException ex) 

10 | { 

11 Console.WriteLine ("Erreur de référence nulle : " + ex); 
12| } 


Dans ce code, cela veut dire que si le code surveillé provoque une FormatException, 
alors nous afficherons le message « Erreur de format... »; s’il provoque une 
NullReferenceException, alors nous afficherons le message « Erreur de référence 
nulle... ». Notez bien que nous ne pouvons rentrer à chaque fois que dans un seul bloc 
catch. Une autre solution serait d'attraper une exception plus généraliste par exemple 
SystemException dont dérive FormatException et NullReferenceException : 
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1| try 

2| € 

3 // code provoquant une exception 

al} 

5| catch (SystemException ex) 

6| 

7 Console.WriteLine ("Erreur système : " + ex); 
8| } 


Par contre, le code précédent attrape toutes les exceptions dérivant de SystemException. 
C’est le cas de FormatException et NullReferenceException, mais c’est aussi le cas 
pour beaucoup d’autres exceptions. Lorsqu'on surveille un bloc de code, on commence 
en général par surveiller les exceptions les plus fines possible et on remonte en consi- 
dérant les exceptions les plus générales, jusqu’à terminer par la classe Exception : 


1| try 

2| € 

3 // code provoquant une exception 

al} 

5| catch (FormatException ex) 

6| 

7 Console.WriteLine("Erreur de format : " + ex); 

8| } 

9| catch (NullReferenceException ex) 

10! 

11 Console.WriteLine("Erreur de référence nulle : " + ex); 

1217} 

13| catch (SystemException ex) 

14| € 

15 Console.WriteLine ("Erreur système autres que 
FormatException et NullReferenceException : " + ex); 

16! } 

17| catch(Exception ex) 

18! 

19 Console.WriteLine ("Toutes les autres exceptions : " + ex); 

20| } 


À chaque exécution, c’est le bloc catch qui se rapproche le plus de l'exception levée 
qui est utilisé. C’est un peu comme une opération conditionnelle. Le programme vérifie 
dans un premier temps que l’exception n’est pas une FormatException. Si ce n’est 
pas le cas, il vérifiera qu'il n’a pas à faire à une NullReferenceException. Ensuite, 
il vérifiera qu'il ne s’agit pas d’une SystemException. Enfin, il interceptera toutes les 
exceptions dans le dernier bloc de code, car Exception étant la classe mère, toutes les 
exceptions sont interceptées avec ce type. 


À noter qu'il est possible d’imbriquer les try...catch si cela s'avère pertinent. Par 
exemple : 


1| string saisie = Console.ReadLine(); 
2| try 
3| À 
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4 int entier = Convert.Tolnt32(saisie); 

5 Console.WriteLine("La saisie est un entier"); 

6| } 

7| catch (FormatException) 

8| À 

9 try 

10 { 

11 double d = Convert.ToDouble(saisie); 

12 Console.WriteLine("La saisie est un double"); 

13 } 

14 catch (FormatException) 

15 { 

16 Console.WriteLine("La saisie n'est ni un entier, ni un 
double"); 

17 x 

18| } 


Ce code nous permet de tester si la saisie est un entier. Si une exception se produit, 
alors nous tentons de le convertir en double. S'il y a encore une exception, alors nous 
affichons un message indiquant que les deux conversions ont échoué. 


Lever une exception 


Il est possible de déclencher soi-même la levée d’une exception. C’est utile si nous 
considérons que notre code a atteint un cas limite, qu’il soit fonctionnel ou technique. 
Pour lever une exception, nous utilisons le mot-clé throw, suivi de l'instance d’une 
exception. Imaginons par exemple une méthode permettant de calculer la racine carrée 
d’un double. Nous pouvons obtenir un cas limite lorsque nous tentons de passer un 
double négatif : 


1| public static double RacineCarree(double valeur) 

2e 

3 if (valeur <= 0) 

4 throw new ArgumentOutOfRangeException("valeur", "Le 
paramètre doit être positif"); 

5 return Math.Sqrt (valeur); 

6| } 


Ici, nous instancions une ArgumentOutOfRangeException en utilisant un constructeur 
à deux paramètres. Ceux-ci permettent d’indiquer le nom du paramètre ainsi que le 
message d'erreur. Cette exception est une exception du framework .NET utilisée pour 
indiquer qu’un paramètre est en dehors des plages de valeurs autorisées. C’est exacte- 
ment ce qu’il nous faut ici. Puis, nous levons l'exception avec le mot-clé throw. Nous 
pouvons utiliser la méthode ainsi : 


1| static void Main(string[] args) 


try 
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4 { 

5 double racine = RacineCarree(-5); 

6 } 

7 catch (Exception ex) 

8 { 

9 Console.WriteLine ("Impossible d'effectuer le calcul : " 
+ ex.Message); 

10 } 

ul} 


Ce qui affiche : 


Impossible d’effectuer le calcul : Le paramètre doit être 


positif 
Nom du paramètre : valeur 


Il aurait été possible de lever une exception plus générique avec la classe de base 
Exception : 


1 | throw new Exception("Le paramètre doit être positif"); 


Mais n'oubliez pas que plus l’exception est finement typée, plus elle sera facile à traiter 
précisément. Cela permet d'éviter d’attraper toutes les exceptions dans le même catch 
avec la classe de base Exception. À noter que lorsque notre programme rencontre le 
mot-clé throw, il s’arrête pour partir dans le bloc try...catch correspondant (s’il 
existe). Cela signifie qu’une méthode qui doit renvoyer un paramètre pourra compiler 
si son chemin se termine par une levée d'exception, comme c’est le cas pour le calcul 
de la racine carrée. 


Propagation de l’exception 


Il est important de noter que lorsqu'un bout de code se situe dans un bloc try...catch, 
tout le code qui est dessous est surveillé, même s’il y a plusieurs méthodes qui s’ap- 
pellent les unes les autres. Par exemple, si nous appelons depuis la méthode Main() 
une Methode1(), qui appelle une Methode2(), qui appelle une Methode3(), qui lève 
une exception, alors nous serons capables de l’intercepter depuis la méthode Main() 
avec un try...catch: 


1| static void Main(string[] args) 

2| € 

3 try 

4 { 

5 Methode); 

6 } 

7 catch (NotIlmplementedException) 

8 { 

9 Console.WriteLine("On intercepte l'exception de la mé 


vhôde 3°): 


343 


CHAPITRE 31. GÉRER LES ERREURS : LES EXCEPTIONS 


10 ; 

ul} 

12 

13| public static void Methodel() 
14] 

15 Methode2(); 

16| } 

17 

18| public static void Methode2() 
19| 

20 Methode3(); 

21| } 

22 

23| public static void Methode3() 
24| 

25 throw new NotImplementedException(); 
26| } 


À noter qu’une NotImplementedException est une exception utilisée pour indiquer 
qu’un bout de code n’a pas encore été implémenté. Il est également possible d’attraper 
une exception, de la traiter et de choisir qu’elle continue à se propager. Par exemple, 
imaginons que nous avons un bloc try...catch qui nous permet de surveiller tout notre 
programme et que nous ayons à surveiller un bout de code ailleurs dans le programme 
qui peut produire une situation limite : 


1| static void Main(string[] args) 

2| € 

3 try 

4 { 

5 MaMethode() ; 

6 } 

7 catch (Exception ex) 

8 { 

9 // ici, on intercepte toutes les erreurs possibles en 
indiquant qu'un problème inattendu s'est produit 

10 Console.WriteLine ("L'application a rencontré un problè 
me, un mail a été envoyé à l'administrateur ..."); 

11 EnvoyerExceptionAdministrateur (ex); 

12 x 

13| } 

14 

15] public static void MaMethode() 

16| 

17 try 

18 { 

19 Console.WriteLine("Veuillez saisir un entier :"); 

20 string chaine = Console.ReadLine() ; 

21 int entier = Convert.Tolnt32(chaine); 

22 + 

23 catch (FormatException) 
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24 { 

25 Console.WriteLine("La saisie n'est pas un entier"); 
26 } 

27 catch (Exception ex) 

28 { 

29 EnregistrerErreurDansUnFichierDeLog(ex); 

30 throv: 

31 } 

32| } 


J’ai une saisie à faire et à convertir en entier. Si la conversion échoue, je suis capable 
de l’indiquer à l'utilisateur en surveillant la FormatException. Par contre, si une ex- 
ception inattendue se produit, je souhaite pouvoir faire quelque chose, en l’occurrence 
enregistrer l'exception dans un fichier ; mais comme c’est un cas limite non prévu je 
souhaite que l’exception continue à se propager afin qu’elle soit attrapée par le bloc 
try...catch qui permettra d'envoyer un mail à l'administrateur. Dans ce cas, j'utilise 
un catch généraliste pour traiter les exceptions inattendues afin de loguer l'exception, 
puis j'utilise directement le mot-clé throw afin de permettre de relever l’exception. 


Créer une exception personnalisée 


Grâce au typage fort des exceptions, il est pratique d'utiliser un type d'exception pour 
reconnaître un cas limite, comme une erreur de conversion ou une exception de référence 
nulle. Nous allons pouvoir utiliser certaines de ces exceptions pour nos besoins, comme 
ce que nous avions fait avec l’exception ArgumentOutOfRangeException. Bien sûr, il 
est possible de créer nous-mêmes nos exceptions afin de lever nos propres exceptions 
correspondant à des cas limites fonctionnels ou techniques. Par exemple, imaginons un 
site d’e-commerce qui affiche une page correspondant au descriptif d’un produit afin de 
pouvoir le commander. Nous chargeons le produit. Si le produit n’est plus en stock alors 
il peut être judicieux de lever une exception afin que le site puisse gérer ce cas limite 
et afficher un message en conséquence. Créons donc notre exception personnalisée : 
ProduitNonEnStockException... 


Pour ce faire, il suffit de créer une classe qui dérive de la classe de base Exception : 


1| public class ProduitNonEnStockException : Exception 
2| € 
3| } 


Qui pourra être utilisée ainsi : 


1| static void Main(string[] args) 

2| € 

3 try 

4 { 

5 ChargerProduit ("TV HD"); 

6 } 

7 catch (ProduitNonEnStockException ex) 
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8 { 

9 Console.WriteLine("Erreur : " + ex.Message); 
10 x 

ul} 

12 


13] public static Produit ChargerProduit (string nomProduit) 
14] 


15 Produit produit = new Produit (); // à remplacer par le 
chargement du produit 

16 if (produit.Stock <= 0) 

17 throw new ProduitNonEnStockException(); 

18 return produit; 

19| } 


Il serait intéressant de pouvoir rendre l’exception plus explicite en modifiant par exemple 
la propriété message de l’exception. Pour ce faire, il suffit d’utiliser la surcharge du 
constructeur prenant une chaîne de caractères en paramètre afin de pouvoir mettre à 
jour la propriété Message (qui est en lecture seule) : 


1| public class ProduitNonEnStockException : Exception 

2| € 

3 public ProduitNonEnStockException() : base("Le produit n' 
est pas en stock") 

4 { 

5 } 


6| } 


Nous pouvons également créer un constructeur qui prend le nom du produit en para- 
mètre afin de rendre le message encore plus précis : 


1| public class ProduitNonEnStockException : Exception 

2| € 

3 public ProduitNonEnStockException(string nomProduit) : base 
("Le produit " + nomProduit + " n'est pas en stock") 

4 L 

5 } 

6| } 


Que nous pourrons utiliser ainsi : 


1| public static Produit ChargerProduit (string nomProduit) 
2| € 
3 Produit produit = new Produit (); // à remplacer par le 
chargement du produit 
if (produit.Stock <= 0) 
throw new ProduitNonEnStockException(nomProduit); 
return produit; 


NN Oo à À 


} 


Ce qui donne une belle levée d'exception : 


Erreur : Le produit TV HD n’est pas en stock 
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À noter que pour construire cette exception personnalisée, nous avons dérivé de la 
classe de base Exception. Il aurait été également possible de dériver de la classe 
ApplicationException pour conserver une hiérarchie cohérente d’exceptions. 


Le mot-clé finally 


Nous avons vu que lorsqu'un code est surveillé dans un bloc try...catch, on sortait 
du bloc try si jamais une exception était levée pour atterrir dans le bloc catch. Cela 
veut dire qu’il est impossible de garantir qu’une instruction sera exécutée dans notre 
code, si jamais une exception nous fait sortir du bloc. C’est là qu’intervient le mot-clé 
finally. Il permet d'indiquer que dans tous les cas, un code doit être exécuté, qu’une 
exception intervienne ou pas. Par exemple : 


1| try 

2| € 

3 string saisie = Console.ReadLine(); 

4 int valeur = Convert.Tolnt32(saisie); 

5 Console.WriteLine("Vous avez saisi un entier"); 

6| } 

7| catch (FormatException) 

8| t 

9 Console.WriteLine("Vous avez saisi autre chose qu'un entier 
DE 

10! } 

11| finally 

121 { 

13 Console.WriteLine("Merci d'avoir saisi quelque chose"); 

141} 


Nous demandons une saisie utilisateur. Nous tentons de convertir cette saisie en entier. 
Si la conversion fonctionne, nous restons dans le bloc try : 


AMI 


Merci d’avoir saisi quelque chose 


Si la conversion lève une FormatException, le texte affiché dans la console sera diffé- 
rent : 


abc 
Vous avez saisi autre chose qu’un entier 


Merci d’avoir saisi quelque chose 


Dans les deux cas, nous passons obligatoirement dans le bloc finally pour afficher le 
remerciement. 


Le bloc finally est utile par exemple quand il s’agit de libérer la mémoire, d’enregistrer 
des données, d'écrire dans un fichier de log, etc. 
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Il est important de remarquer que quelle que soit la construction, le bloc finally est 
toujours exécuté. Ainsi, même si on relève une exception dans le bloc catch : 


1| try 

2| € 

3 Convert.Tolnt32("ppp"); 

al} 

5| catch (FormatException) 

6| 

7 throw new NotImplementedException(); 
8|} 

9| finally 

10| 

11 Console.WriteLine("Je suis quand même passé ici"); 
12| } 


nous afficherons toujours notre message... 


Même lorsque nous souhaitons sortir d’une méthode avec le mot-clé return : 


1| static void Main(string[] args) 
2| € 

3 MaMethode() ; 

al} 

5 

6| private static void MaMethode() 
7T|t 

8 try 

9 { 

10 Convert.Tolnt32("ppp"); 
11 } 

12 catch (FormatException) 

13 { 

14 return; 

15 } 

16 finally 

17 { 

18 Console.WriteLine("Je suis quand même passé ici"); 
19 } 

20| } 


Ici, le bloc finally est quand même exécuté. 


Dans ce chapitre, nous avons vu comment gérer les cas limites en utilisant le mécanisme 
des exceptions. Vous avez pu voir qu'il est très pratique de gérer les exceptions grâce à 
leurs types et qu’il est finalement très facile de créer nos propres exceptions. 


Attention : la gestion des exceptions ne doit pas masquer les bugs. Vous ne devez pas 
entourer tout votre code de blocs try...catch et vous féliciter qu’il n’y ait aucun bug. 
C’est au contraire une opportunité pour les identifier et produire un rapport pour les 
corriger ultérieurement, par exemple en envoyant un e-mail avec le contenu du bug aux 
développeurs. 
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En résumé 


— Les exceptions permettent de gérer les cas limites d’une méthode. 

— On utilise le bloc try...catch pour encapsuler un bout de code à surveiller. 

Il est possible de créer des exceptions personnalisées en dérivant de la classe de base 
Exception. 

On peut lever à tout moment une exception grâce au mot-clé throw. 

— Les exceptions ne doivent pas servir à masquer les bugs. 
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TP : événements et météo 


Difficulté : 


ienvenue dans le dernier TP de cette partie. Tenez bon, après on change de domaine 
B pour aborder d'autres notions. Dans ce TP, nous allons pratiquer les événements. 

Le but est de savoir en créer un et de pouvoir s'y abonner pour être notifié d'une 
information. 


Vous êtes prêts ? Alors c'est parti! 


391 


CHAPITRE 32. TP : ÉVÉNEMENTS ET MÉTÉO 


Instructions pour réaliser le TP 


Nous allons réaliser ici un mini simulateur de météo qui sera utilisé par un statisticien 
afin d’en faire des statistiques (logique !). Bon, c’est le contexte, mais c’est juste un 
exemple. N’espérez pas non plus réaliser un vrai simulateur météo dans ce TP! 


Bref : nous devons créer un simulateur de météo. Lorsque celui-ci est démarré, il génère 
autant de nombres aléatoires que demandé, des nombres entre 0 et 100. Si le nombre 
aléatoire est inférieur à 5, alors ça veut dire que le temps est au soleil. S’il est supérieur 
ou égal à 5 et inférieur à 50, alors nous aurons des nuages. S’il est supérieur ou égal à 
50 et inférieur à 90, alors nous aurons de la pluie. Sinon, nous aurons de l’orage. Un 
événement sera levé à chaque changement de temps. Le but de notre statisticien est de 
s'abonner aux événements du simulateur météo afin de compter le nombre de fois où il 
a fait soleil et le nombre de fois où le temps a changé. Il affichera ensuite son rapport 
en indiquant ces deux résultats et le pourcentage de fois où il a fait soleil (je veux bien 
que ce pourcentage soit un entier). 


C’est tout pour l'énoncé ! Maintenant, vous avez assez de connaissances pour que je ne 
détaille pas plus. Bon courage! 


Correction 


Allez, c’est parti pour la correction. Tout d’abord, nous devons créer notre simulateur 
météo. Il sera représenté par une classe : 


1| public class SimulateurMeteo 
2| € 
3| } 


Nous aurons également besoin de quelque chose pour représenter le temps : soleil, 
nuage, pluie et orage. Une énumération semble appropriée : 


1| public enum Temps 
2| € 

3 Soleil, 

4 Nuage, 

5 Pluie, 

6 Orage 

7|} 


Enfin, nous aurons notre statisticien : 


1| public class Statisticien 
2| € 
3[} 


Commençons par le simulateur de météo. Nous aurons besoin de plusieurs variables 
membres afin de stocker notre générateur de nombres aléatoires, le dernier temps qu’il 
a fait et le nombre de répétitions. Le nombre de répétitions pourra être un paramètre 
du constructeur : 
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public class SimulateurMeteo 


L 


} 


private Temps? ancienTemps; 
private int nombreDeRepetitions; 
private Random random; 


public SimulateurMeteo(int nombre) 


{ 
random = new Random); 
ancienTemps = null; 
nombreDeRepetitions = nombre; 
} 


Étant donné que nous allons déterminer plusieurs nombres aléatoires, il est pertinent de 
ne pas réallouer à chaque fois le générateur de nombre aléatoire. C’est pour cela qu’on le 
crée une unique fois dans le constructeur de la classe. Créons maintenant une méthode 
permettant de démarrer le simulateur et codons les règles métier du simulateur : 


© © I OO Où À À ND 


public class SimulateurMeteo 


{ 


//T...] Code supprimé pour plus de clarté [...] 


public void Demarrer() 


{ 
for (int i = 0; i < nombreDeRepetitions; i++) 
{ 
int valeur = random.Next (0, 100); 
if (valeur < 5) 
GererTemps (Temps.Soleil); 
else 
GA 
if (valeur < 50) 
GererTemps (Temps. Nuage) ; 
else 
{ 
if (valeur < 90) 
GererTemps (Temps.Pluie); 
else 
GererTemps (Temps .0rage); 
} 
} 
} 
} 


C’est très simple, on boucle sur le nombre de répétitions. Un nombre aléatoire est 
déterminé à chaque itération. La méthode GererTemps prend en paramètre le temps 
déterminé à partir du nombre aléatoire. C’est cette méthode GererTemps qui aura pour 
but de lever un événement quand le temps change : 
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1| public class SimulateurMeteo 

2| { 

3 public delegate void IlFaitBeauDelegate (Temps temps); 

4 public event IlFaitBeauDelegate QuandLeTempsChange; 

5 

6 // T...] Code supprimé pour plus de clarté [...] 

7 

8 private void GererTemps (Temps temps) 

9 { 

10 if (ancienTemps.HasValue && ancienTemps.Value != temps 
&& QuandLeTempsChange != null) 

11 QuandLeTempsChange (temps) ; 

12 ancienTemps = temps; 

13 } 

ul} 


Ici, j'ai choisi de créer un seul événement quand le temps change et de lui indiquer le 
temps qu’il fait en paramètre. Nous avons donc besoin d’un délégué qui prend un Temps 
en paramètre et qui ne renvoie rien - c’est d’ailleurs souvent le cas des événements. Puis 
nous avons besoin d’un événement du type du délégué. Ensuite, si le temps a changé 
et que quelqu'un s’est abonné à l'événement, alors nous levons l’événement. 


Il ne reste plus qu’à remplir notre classe Statisticien. Cette classe a besoin de tra- 
vailler sur une instance de la classe SimulateurMeteo; nous pouvons donc lui en pas- 
ser une dans les paramètres du constructeur. Nous utiliserons également des variables 
membres privées permettant de stocker ses analyses : 


1| public class Statisticien 

2| € 

3 private SimulateurMeteo simulateurMeteo; 

4 private int nombreDeFoisOuLeTempsAChange; 

5 private int nombreDeFoisOuIlAFaitSoleil; 

6 

7 public Statisticien(SimulateurMeteo simulateur) 

8 { 

9 simulateurMeteo = simulateur; 

10 nombreDeFoisOuLeTempsAChange = 0; 

ai nombreDeFoisQuIlAFaitSoleil = O0; 

12 } 

13 

14 public void DemarrerAnalyse() 

15 { 

16 simulateurMeteo.QuandLeTempsChange += 
simulateurMeteo_QuandLeTempsChange; 

17 simulateurMeteo.Demarrer(); 

18 } 

19 

20 public void AfficherRapport () 

21 { 

22 Console.WriteLine ("Nombre de fois où le temps a changé 


"+ nombreDeFoisOuLeTempsAChange); 
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23 Console.WriteLine ("Nombre de fois où il a fait soleil 
" + nombreDeFoisOullAFaitSoleil); 
24 Console.WriteLine("Pourcentage de beau temps : " + 


nombreDeFoisOullAFaitSoleil * 100 / 
nombreDeFoisOuLeTempsAChange + " #"); 

25 } 

26 

27 private void simulateurMeteo_QuandLeTempsChange (Temps temps 


) 
28 À 
29 if (temps == Temps.Soleil) 
30 nombreDeFoisOuIllAFaitSoleil++; 
31 nombreDeFoisOuLeTempsAChange++; 
32 } 
33| } 


Notons que dans la méthode DemarrerAnalyse, nous nous abonnons à l’événement de 
changement de temps. La méthode qui est appelée lors de la notification est très simple, 
elle incrémente les compteurs. Enfin, l'affichage du rapport est trivial. Ici, comme nous 
n'avons que des entiers, la division produira un entier également. I ne reste plus qu’à 
faire fonctionner nos objets : 


1| class Program 

2! { 

3 static void Main(string[]l args) 

4 { 

5 SimulateurMeteo simulateurMeteo = new SimulateurMeteo( 
1000) ; 

6 Statisticien statisticien = new Statisticien( 
simulateurMeteo); 

7 statisticien.DemarrerAnalyse(); 

8 statisticien.AfficherRapport(); 

9 } 

10! } 


Ici, je travaille sur 1000 répétitions. Voici le résultat que j'obtiens à l’exécution de 
l’application : 


Nombre de fois où le temps a changé : 646 


Nombre de fois où il a fait soleil : 55 
Pourcentage de beau temps : 8 % 


Evidemment, vu que nous travaillons avec des nombres aléatoires, chacun aura un 
résultat différent. 


Et voilà, c’est terminé pour ce TP. Notre application est fonctionnelle ! 


Terminé ? hmm... pas tout à fait, allons un peu plus loin. 
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Aller plus loin 


En l’état, ce code est fonctionnel. C’est parfait. Mais que se passe-t-il si nous démarrons 
plusieurs fois l’analyse ? Nous pouvons essayer : 


1| static void Main(string[] args) 


2| { 

3 SimulateurMeteo simulateurMeteo = new SimulateurMeteo(1000) 
; 

4 Statisticien statisticien = new Statisticien( 
simulateurMeteo); 

5 statisticien.DemarrerAnalyse(); 

6 statisticien.AfficherRapport(); 

Né 

8 statisticien.DemarrerAnalyse(); 

9 statisticien.AfficherRapport(); 

10 

11 statisticien.DemarrerAnalyse(); 

12 statisticien.AfficherRapport(); 

13| } 


Ce qui affichera : 


Nombre de fois où le temps a changé : 635 
Nombre de fois où il a fait soleil : 47 
Pourcentage de beau temps : 7 % 

Nombre de fois où le temps a changé : 1917 
Nombre de fois où il a fait soleil : 163 
Pourcentage de beau temps : 8 % 

Nombre de fois où le temps a changé : 3879 
Nombre de fois où il a fait soleil : 322 


Pourcentage de beau temps : 8 % 


Les valeurs augmentent... alors qu’elles ne devraient pas. Eh oui, nous ne réinitialisons 
pas les entiers permettant de stocker les statistiques. Ce n’est pas forcément un bug. 
Ici, comme je n’avais rien dit dans l’énoncé, il peut paraître pertinent de continuer à 
incrémenter ces valeurs, comme ça, je peux travailler sur une moyenne. Bon, disons 
que nous souhaitons les réinitialiser à chaque fois ; il suffit alors de remettre à zéro les 
compteurs dans la méthode : 


1| public void DemarrerAnalyse() 

2| € 

3 nombreDeFoisOuLeTempsAChange = 0; 

4 nombreDeFoisOuIlAFaitSoleil = O; 

5 simulateurMeteo.QuandLeTempsChange += 
simulateurMeteo_QuandLeTempsChange; 

6 simulateurMeteo.Demarrer); 

7|} 


Cependant, les compteurs augmentent toujours, moins vite, mais quand même ! Je suis 
sûr que vous l'aviez deviné et que vous n’avez pas fait l'erreur. En fait, cela vient 
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de l’abonnement à l'événement. Chaque fois que nous démarrons l’analyse, nous nous 
réabonnons à l’événement. Comme l’événement est multidiffusion, nous rajoutons en 
fait à chaque fois un appel à la méthode avec le +—. Ce qui veut dire qu’à la deuxième 
fois, nous appellerons la méthode deux fois, ce qui fera doubler les résultats. À la 
troisième fois, ils triplent... Nous avons donc une erreur. Soit il faut s'abonner une 
seule fois à l'événement, soit il faut se désabonner à la fin de l’analyse, quand ce 
n’est plus utile de recevoir l'événement. C’est cette dernière méthode que je vais vous 
montrer. Il suffit de faire : 


1| public void DemarrerAnalyse() 

2| { 

3 nombreDeFoisOuLeTempsAChange = 0; 

4 nombreDeFoisOuIllAFaitSoleil = O; 

5 simulateurMeteo.QuandLeTempsChange += 


simulateurMeteo_QuandLeTempsChange; 
simulateurMeteo.Demarrer(); 
7 simulateurMeteo.QuandLeTempsChange -= 
simulateurMeteo_QuandLeTempsChange; 


[e)] 


8| } 


On utilise l’opérateur -— pour enlever la méthode du délégué. 


D'une manière générale, il est bienvenu de se désabonner d'un événement 
lorsque l'on sait qu'on ne va plus s'en servir. 


Cela permet d’éviter d’encombrer la mémoire qui ne saurait pas forcément se libérer 
toute seule. Je n’en dis pas plus car ceci est un concept avancé de gestion de mémoire. 
Gardez seulement à l’esprit que si on a l’opportunité de se désabonner d’un événement, 
il faut le faire. 


Enfin, nous pouvons simplifier notre code en ne créant pas notre délégué. Effectivement, 
dans la mesure où celui-ci possède un seul paramètre, il est possible de le remplacer par 
le délégué Action<T>. Il faut juste supprimer la déclaration du délégué et remplacer la 
déclaration de l’événement par : 


1| public class SimulateurMeteo 

2| € 

3 public event Action<Temps> QuandLeTempsChange; 
al} 


Vous pouvez télécharger tous les codes sources de cet exercice grâce au code web sui- 
vant : 


Copier ce code ) 


Code web : 179137 
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C- avancé 
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Créer un projet bibliothèques de classes 


Difficulté : 16: 


our l'instant, nous n'avons créé que des projets de type application console. Il existe 

plein d'autres modèles de projet. Un des plus utiles est le projet permettant de créer des 

bibliothèques de classes. Il s'agit d'un projet qui va permettre de contenir des classes 
que nous pourrons utiliser dans des applications. Exactement comme la bibliothèque de 
classes du framework .NET qui nous permet d'utiliser la classe Math ou les exceptions, 
ou plein d’autres choses... Ce type de projet permet donc de créer une assembly sous la 
forme d'un fichier avec l'extension .d11. Ces assemblys sont donc des bibliothèques qui 
seront utilisables par n'importe quelle application utilisant un langage compatible avec le 
framework .NET. 
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Pourquoi créer une bibliothèque de classes ? 


Globalement pour deux raisons que nous allons détailler : 
— Réutilisabilité. 
— Architecture. 


Réutilisabilité : 


Comme indiqué en introduction, le projet de type bibliothèque de classes permet d’ob- 
tenir des assemblys avec l'extension .d11. Nous pouvons y mettre tout le code C# que 
nous voulons, notamment des classes qui auront un intérêt à être utilisées à plusieurs 
endroits ou partagées par plusieurs applications. C’est le cas des assemblys du frame- 
work .NET. Elles possèdent plein de code très utile que nous aurons avantage à utiliser 
pour créer nos applications. 


De la même façon, nous allons pouvoir créer des classes qui pourront être réutilisées à 
plusieurs endroits. Comme notre classe permettant de gérer les listes chaînées. N’im- 
porte quel programme qui utilise des listes chaînées aura intérêt à ne pas tout réinventer 
et à simplement réutiliser ce code prêt à l’emploi. Il suffira pour ce faire de réutiliser 
cette classe en référençant l’assembly, comme nous l’avons déjà vu. 


Architecture : 


L'autre avantage dans la création de bibliothèques de classes est de pouvoir archi- 
tecturer son application de manière à faciliter sa mise en place, sa maïintenabilité et 
l’évolutivité du code. L'architecture informatique c’est comme l’architecture d’une mai- 
son. Si nous mettons la douche au même endroit que le compteur électrique, ou que la 
machine à laver est juste à côté de la chambre à coucher, cela peut poser des problèmes. 
De même, que penser d’un architecte qui place les toilettes à côté du lit ? Une maison 
mal architecturée est une maison où il ne fait pas bon vivre. De même, une applica- 
tion mal architecturée est une application dont il ne fait pas bon faire la maintenance 
applicative ! 


Architecturer son application est important surtout si l'application est grosse. Par 
exemple, une bonne pratique dans une application informatique est la décomposition 
en couches. Nous n’allons pas faire ici un cours d’architecture, mais le but est de 
séparer les composantes de l’application. Un grand nombre d’applications de gestion est 
composé d’une couche de présentation, d’une couche métier et d’une couche d’accès aux 
données, les couches communiquant entre elles. Le but est de limiter les modifications 
d’une couche lors de la modification d’une autre. Si toutes les couches étaient mélangées 
alors chaque modification impliquerait une série de modifications en chaîne. On peut 
faire une analogie avec un plat de lasagnes et un plat de spaghettis. Il est difficile de 
toucher à un spaghetti sans toucher les autres. Cependant, il pourrait être envisageable 
de toucher à la couche du dessus du plat de lasagnes pour rajouter un peu de fromage 
sans perturber ce qu’il y a dessous. Miam ! 


Il est donc intéressant d’avoir une bibliothèque de classes qui gère l’accès aux données, 
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une autre qui gère les règles métier et une autre qui s’occupe d’afficher le tout. Je 
m'arrête là sur l’architecture, vous aurez l’occasion de vous y confronter bien assez 
tôt ! 


Créer un projet de bibliothèque de classe 


Pour créer une bibliothèque de classes, on utilise l’assistant de création de nouveau 
projet (Fichier > Nouveau > Nouveau projet), comme on l’a fait pour une appli- 
cation console sauf qu'ici, on utilisera le modèle Bibliothèque de classes (voir la 
figure 33.1). Ne le faisons pas encore. 


= 


Nouveau projet EArx 
Trier par: [Par défaut +] :: Rechercher Modèles installés ? 
Type: Visual C# 
| =c#] Application Windows Forms Visual C# 
E ppi Windows F. isual C ype: Visu: 
Visual C# = Projet de création d'une bibliothèque de 
classes C# (dll) 
Modèles en ligne | <! Application WPF Visual C# 
El Application console Visual C# 
Le Bibliothèque de classes Visual C# 
m= se . : 
Fc] Application de navigateur WPF Visual C# 
CA] Projet vide Visual C# 


Nom: MaBibliothequel 


FIGURE 33.1 - Choix d’un projet de type Bibliothèque de classes 


Si nous faisons cela, Visual C# Express va nous créer une nouvelle solution contenant 
le projet de bibliothèques de classes. C’est possible sauf qu’en général, on ajoute une 
bibliothèque de classes pour qu’elle serve dans le cadre d’une application. Cela veut dire 
que nous pouvons ajouter ce nouveau projet à notre solution actuelle. Aïnsi, celle-ci 
contiendra deux projets : 


— L'application console, qui sera exécutable par le système d’exploitation. 
— La bibliothèque de classes, qui sera utilisée par l’application. 


Pour ce faire, on peut faire un clic droit sur notre solution et faire Ajouter > Nouveau 
Projet, comme indiqué à la figure 33.2. 


Ici, on choisit le projet de type bibliothèque de classes, sans oublier de lui donner un 
nom, par exemple MaBibliotheque (voir figure 33.3). Visual C4} Express nous crée 
notre projet et l’ajoute à la solution, comme on peut le voir dans l’explorateur de 
solutions (voir figure 33.4). 
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“Solution 'MaPremiereApplication' (1 projet) | 


ES Générer la solution F6 MaPremiereApplicati 
Régénérer la solution 5 Properties 


Définir les projets de démarrage... 
fh Coller Ctri+V 
Renommer 


Propriétés 


pauses Eu 
del Rechercher Modëles installés 
es — : + 
El Application Windows Forms Visual C# Los 
Visual C# = Projet de création d'une bibliothèque de 
classes C# (dl) 


lication WPF Visual C# 
pl 


Modèles en ligne il 
em Application console Visual C# 


Bibliothèque de classes Visual C# 

c4| Application de navigateur WPF Visual C# 
En 

Projet vide Visual C# 


heque 


wremlocuments\visual studio 2010\Projects\C#\MaPremiereApplication È 


Nom: 


Emplacement : 


FIGURE 33.3 — Création d’un projet Bibliothèque de classes appelé MaBibliotheque 


Explorateur de solutions 


LAINE 
Cd Solution 'MaPremiereApplication' (2 projets) 
4 KL] MaBibliotheque 
b (4 Properties 
b Le Références 
%] Classl.cs 
4 (fl MaPremiereApplication 
b [5] Properties 
b La Références 
#] Program.cs 


FIGURE 33.4 — L’explorateur de solution affiche les deux projets contenus dans la 
solution 


364 


PROPRIÉTÉS DE LA BIBLIOTHÈQUE DE CLASSE 


Il le génère avec un fichier Classi.cs que nous pouvons supprimer. 
Remarquez que l’application console est en gras. Cela veut dire que c’est ce projet que 
Visual C# Express va tenter de démarrer lorsque nous utiliserons les touches 


+ (F5) Si ce n’est pas le cas, il peut tenter de démarrer la bibliothèque, ce qui n’a pas 
de sens vu qu’elle n’a pas de méthode Main(). Dans ce cas, vous pourrez le changer 
en cliquant droit sur le projet console et en choisissant Définir comme projet de 
démarrage. 


Dans ce projet, nous pourrons désormais créer nos classes en ajoutant de nouvelles 
classes au projet, comme on l’a déjà fait avec l’application console. 


Ajoutons par exemple la classe suivante : 


1| namespace MaBibliotheque 

2| € 

3 public class Bonjour 

4 { 

5 public void DireBonjour() 

6 Le 

7 Console.WriteLine("Bonjour depuis la bibliothèque 
MaBibliotheque"); 

8 } 

9 } 

10| } 


Propriétés de la bibliothèque de classe 


Ouvrons la fenêtre des propriétés de notre projet, en faisant un clic droit sur le projet, 
puis Propriétés (voir la figure 33.5). 


Fichier Edition Affichage Projet Déboguer Données Outils Fenêtre ? 
iJd-Sdd#i43m|9-c-|r|æ 


ù 


Application 
Las Nom de l'assembly : Espace de noms par défaut : 
Générer MaBibliotheque 


sainos gs sjno e 210g 


Événements de build Framework cible : Type de sortie: 
7 [NET Framework 4 +] [Bibliothèque de classes -] 

Déboguer 
a Objet de démarrage : 
& Ressources = 
3 [ {Non défini) = | Informations de l'assembly... | 
ê& Paramètres 

Ressources 


Chemins d'accès des références Ë C Le n 
Spécifiez comment les ressources de l'application sont gérées : 


Signature 
@ Icône et manifeste 


Un manifeste détermine les paramètres spécifiques pour une application. Pour incorporer un 
TS RS 


EE + = ET 


FIGURE 33.5 — Le nom de l’assembly et l’espace de noms par défaut dans les propriétés 
du projet 


Nous pouvons voir différentes informations et notamment dans l’onglet Application 
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que le projet possède un nom d’assembly, qui correspond ici au nom du projet ainsi 
qu’un espace de noms par défaut, qui correspond également au nom du projet. Le nom 
de l’assembly servira à identifier l’assembly, alors que l’espace de noms par défaut sera 
celui donné lorsque nous ajouterons une nouvelle classe (ou autre) à notre projet. Cet 
espace de noms pourra être changé, mais en général on garde un nom cohérent avec les 
arborescences de notre projet. 


Notons au passage qu’il y a une option permettant d'indiquer la version du framework 
.NET que nous souhaitons utiliser. Ici, nous gardons la version 4. 


Générer et utiliser une bibliothèque de classe 


Nous pouvons alors compiler cette bibliothèque de classes, soit individuellement, soit 
en compilant tout le projet. 


Rendons-nous dans le répertoire où nous avons sauvegardé notre bibliothèque. Nous 
voyons dans le répertoire de sortie (chez moi : C:\Users\Nico-PC\Documents\Visual 
Studio 2010\Projects\C#\MaPremiereApplication\MaBibliotheque\bin\Release) 
qu'il y a un fichier du nom de notre projet dont l’extension est .d11. C’est notre bi- 
bliothèque de classes (même s’il n’y a qu’une classe dedans !). Elle possède le même 
nom que celui que nous avons vu dans les propriétés du projet. 


J'en profite pour vous faire remarquer qu’à son côté, il y a également un fichier du 
même nom avec l’extension .pdb. Je peux enfin vous révéler de quoi il s’agit. Ce fichier 
contient les informations de débogages, utiles lorsque nous déboguons notre application. 
Sans ce fichier, il est impossible de déboguer à l’intérieur du code de notre classe. 


Revenons à l’assembly générée. C’est cette 411 qu’il faudra référencer dans notre projet, 
comme nous l’avons vu au chapitre sur le référencement d’assemblys. Si vous avez un 
doute, n'hésitez pas à retourner le consulter. Aïnsi, nous serons en mesure d'utiliser la 
classe de notre bibliothèque. Rappelez-vous, pour référencer une assembly, nous faisons 
un clic droit sur les références du projet et sélectionnons Ajouter une référence. 
Nous pourrons référencer notre bibliothèque de deux manières ; soit en utilisant l’onglet 
Parcourir, qui va nous permettre de pointer directement le fichier d11 contenant nos 
classes (voir figure 33.6) ; soit en référençant directement le projet de bibliothèque de 
classes qui se trouve dans notre solution en utilisant l’onglet Projet (voir figure 33.7). 


C’est ce choix qui doit être privilégié lorsque notre solution contient les projets à 
référencer. Généralement, vos bibliothèques vont évoluer en même temps que votre 
programme, donc il est judicieux de les avoir dans la même solution que l’application. 
Nous pourrons donc faire des références projet. Si cependant les bibliothèques sont 
stables et ne sont pas amenées à évoluer, alors vous pourrez les référencer directement 
à partir des d11; vous y gagnerez en temps de compilation. Utilisons désormais notre 
classe avec le code suivant : 


1| MaBibliotheque.Bonjour bonjour = new MaBibliotheque.Bonjour (); 
2| bonjour.DireBonjour (); 
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MaBibliotheque 
Types de fichiers : [Fichiers composants Cdi” tlb;”.olb;".0cx;".exe;" manifest) 


C0.) Canne] 


FIGURE 33.6 — Référencement direct de l’assembly 


Parcourir | Récent 


| Répertoire du projet 


FIGURE 33.7 — Référencement du projet 
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Vous pouvez aussi, bien sûr, inclure l’espace de noms MaBibliotheque pour éviter 
d’avoir à préfixer la classe. En général, l’espace de noms d’une bibliothèque est différent 
de celui de l’application. Notez que pour qu’une classe, comme la classe Bonjour, puisse 
être utilisée par une application référençant son assembly, elle doit être déclarée en 
public afin qu’elle soit visible par tout le monde. 


Le mot-clé internal 


Nous avons déjà vu ce mot-clé lorsque nous avons parlé de visibilité avec notamment 
les autres mots-clés public, private et protected. 


C’est avec les assemblys que le mot-clé internal prend tout son sens. Il permet d’in- 
diquer qu’une classe, méthode ou propriété ne sera accessible qu’à l’intérieur d’une 
assembly. Cela permet par exemple qu’une classe soit utilisable par toutes les autres 
classes de cette assembly mais qu’elle ne puisse pas être utilisée par une application 
référençant l’assembly. 


Créons par exemple dans notre bibliothèque de classes les trois classes suivantes : 


1| public class Client 

à | 

3 private string login; 

4 public string Login 

5 { 

6 get Ÿ{ return login; } 

dé } 

8 

9 private string motDePasse; 

10 public string MotDePasse 

11 { 

12 get { return motDePasse.Crypte(); } 
13 } 

14 

15 public Client(string loginClient, string motDePasseClient) 
16 { 

17 login = loginClient; 

18 motDePasse = motDePasseClient; 

19 } 

20| } 

21 

22| public static class Generateur 

23 | À 

24 public static string ObtenirldentifiantUnique() 
25 { 

26 Random r = new Random); 

27 string chaine = string.Empty; 

28 for (int i = 0; i < 10; i++) 

29 { 

30 chaine += r.Next (0, 100); 
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31 } 

32 return chaine.Crypte(); 

33 } 

34 } 

35 

36| internal static class Encodage 

37| 

38 internal static string Crypte(this string chaine) 

39 { 

40 return Convert.ToBase64String (Encoding.Default.GetBytes 
(chaine) ); 

41 } 

42 

43 internal static string Decrypte(this string chaine) 

44 € 

45 return Encoding.Default.GetString (Convert. 
FromBase64String(chaine)); 

46 } 

at | } 


Vous pouvez copier ce code en utilisant le code web suivant : 


Copier ce code ) 


Code web : 888787 


Les deux premières sont des classes publiques qui peuvent être utilisées depuis n’importe 
où, par exemple depuis le programme principal : 


1] class Program 

2| € 

3 static void Main(string[l]l args) 

4 { 

5 Client client = new Client ("Nico", "12345"); 

6 Console.WriteLine(client.MotDePasse); 

7 

8 Console.WriteLine (Generateur.0btenirIidentifiantUnique() 
); 

9 } 

10! } 


Par contre, la classe Encodage n’est accessible que pour les deux autres classes, car 
elles sont dans la même assembly. Cela veut dire que si nous tentons de l'utiliser depuis 
notre méthode Main(), nous aurons des erreurs de compilation : 


1| static void Main(string[] args) 

2| € 

3 string chaine = "12345".Crypte(); 
4 

5 Encodage.Crypte("12345"); 

6| } 
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Cet exemple permet d'illustrer l’intérêt, qui n’est pas toujours évident, du mot-clé 
internal. 


À noter qu’il existe enfin le mot-clé protected internal qui permet d'indiquer que des 
éléments sont accessibles à un niveau internal pour les classes d’une même assembly 
mais protected pour les autres assemblys, ce qui permet d’appliquer les principes de 
substitutions ou d’héritage. 


Voilà, vous avez vu comment créer une bibliothèque de classes ! N’hésitez pas à créer ce 
genre de projet afin d’y mettre toutes les classes qui peuvent être utilisées par plusieurs 
applications. Comme ça, il suffit d’une simple référence pour accéder au code qui y est 
contenu. Vous vous en servirez également pour mieux architecturer vos applications, le 
code s’en trouvera plus clair et plus maintenable. 


ce mot-là sur internet. Vous le verrez également mal francisé en « librairie », 
ce qui est évidemment une erreur! 


@ La traduction en anglais de bibliothèque est « library », vous verrez souvent 


En résumé 


— Un projet de bibliothèque de classes permet de regrouper des classes pouvant être 
utilisées entre plusieurs applications. 

— Les bibliothèques de classes génèrent des assemblys avec l’extension .d11. 

— Elles permettent également de mieux architecturer un projet. 
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se D 


Plus loin avec le C# et .NET 


Difficulté : 


aintenant que nous en savons plus, nous allons pouvoir aborder quelques notions qui 

me paraissent importantes et que nous n'avons pas encore traitées. Ce sera l'occasion 

d'approfondir nos connaissances et de comprendre un peu mieux certains points qui 
auraient pu vous paraître douteux. 


Nous allons voir des instructions C# supplémentaires qui vont nous permettre de com- 
pléter nos connaissances en POO. Nous en profiterons également pour détailler comment 
le framework .NET gère les types en mémoire. Vous en saurez plus sur le formatage des 
chaînes et aurez un aperçu du côté obscur du framework .NET, la réflexion ! 


Bref, tout plein de nouvelles cordes à nos arcs nous permettant d'être de plus en plus 
efficace avec le C#. Ces nouveaux concepts vous serviront sans doute moins souvent, mais 
sont importants à connaître. 
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Empêcher une classe de pouvoir être héritée 


Parmi ces nouveaux concepts, nous allons voir comment il est possible de faire en sorte 
qu'une classe ne puisse pas être héritée. Rappelez-vous, à un moment, j’ai dit qu’on ne 
pouvait pas créer une classe qui spécialise la classe String. 


C'est quoi ce mystère ? Nous, quand nous créons une classe, n'importe qui 
peut en hériter. Pourquoi pas la classe String ? 


C’est parce que je vous avais caché le mot-clé sealed. Il permet d'empêcher la classe 
d’être héritée. Tout simplement ! 


Pourquoi vouloir empêcher d'étendre une classe en la dérivant ? Pour plusieurs raisons, 
mais généralement nous allons recourir à ce mot-clé lorsqu'une classe est trop spécifique 
à une méthode ou à une bibliothèque et que l’on souhaite empêcher quelqu'un de 
pouvoir obtenir du code instable en étendant son fonctionnement. C’est typiquement 
le cas pour la classe String. Il y a beaucoup de classes dans le framework .NET qui 
sont marquées comme sealed afin d'éviter que l’on puisse faire n’importe quoi. 


Il faut par contre faire attention car l’emploi de ce mot-clé restreint énormément la 
façon dont on pourrait employer cette classe. Pour déclarer une classe sealed ! il suffit 
de faire précéder le mot-clé class du mot-clé sealed : 


1| public sealed class ClasselmpossibleADeriver 
2| € 
3[ } 


Aiünsi, toute tentative d’héritage : 


1| public class MaClasse : ClasselmpossibleADeriver 
2| € 
3| } 


se soldera par une erreur de compilation déshonorante : 


impossible de dériver du type sealed ?’MaPremiereApplication. 


ClasselmpossibleADeriver ? 


Le mot-clé sealed fonctionne également pour les méthodes, dans ce cas, cela permet 
d'empêcher la substitution d’une méthode. Reprenons notre exemple du chien muet : 


1| public class Chien 

2| { 

3 public virtual void Aboyer() 
4 { 

5 Console.WriteLine ("Wouf"); 
6 } 

7|} 


1. Ce qui signifie « scellée », en anglais. 
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8 

9| public class ChienMuet : Chien 

10! 

11 public sealed override void Aboyer() 
12 { 

13 Console.WriteLine("..."); 

14 } 

15| } 


Ici, nous marquons la méthode comme sealed, ce qui fait qu’une classe qui dérive de 
ChienMuet ne sera pas capable de remplacer la méthode permettant d’aboyer. Le code 
suivant : 


1| public class ChienAvecSyntheseVocale : ChienMuet 
2| € 

3 public override void Aboyer() 

4 { 

5 Console.WriteLine("Bwarf"); 

6 } 

7|} 


entraînera l'erreur de compilation : 


MaPremiereApplication.ChienAvecSyntheseVocale.Aboyer()? 


peut pas se substituer à un membre hérité ? 
MaPremiereApplication.ChienMuet.Aboyer()’?, car il est sealed 


Voilà pour ce mot-clé, à utiliser en connaissance de cause ! 


Précisions sur les types et gestion mémoire 


Nous avons vu beaucoup de types différents au cours des chapitres précédents. Nous 
avons vu les structures, les classes, les énumérations, les délégués, etc. Certains sont 
des types valeur et d’autres des types référence ; ils sont tous des objets. 


Voici à la figure 34.1 un petit schéma récapitulatif des types que nous avons déjà vus. 


Nous avons dit brièvement qu'ils étaient gérés différemment par le framework .NET et 
que les variables de type valeur contenaient directement la valeur, alors que les variables 
de type référence possèdent un lien vers un objet en mémoire. Ce qui se passe en fait 
quand nous déclarons une variable de type valeur, c’est que nous créons directement la 
valeur en mémoire dans ce qu’on appelle « la pile ». C’est une zone mémoire (limitée) 
où il est très rapide de mettre et de chercher des valeurs. De plus, cette mémoire n’a 
pas besoin d’être libérée par le ramasse-miettes (que nous verrons un peu plus loin). 
C’est pour cela que pour des raisons de performances on peut être amené à choisir les 
types valeur. Par exemple lorsque nous faisons ce code : 


1| int age = 10; 
2| double pourcentage = 5.5; 
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objets 


type valeur type référence 


structure 


classe 


interface 


énumération délégué 


FIGURE 34.1 — Récapitulatif des types 


Voici ce qu’il se passe : la variable age est une adresse mémoire contenant la valeur 10. 
C’est la même chose pour la variable pourcentage sauf que l'emplacement mémoire 
est plus important car on peut stocker plus de choses dans un double que dans un int. 
Je vous renvoie à la figure 34.2 pour une explication en images! En ce qui concerne 
les types référence, lorsque nous instancions par exemple une classe, le C# instancie 
la variable maVoiture dans la pile et lui met dedans une référence vers le vrai objet 
qui lui est alloué sur une zone mémoire de capacité plus importante, mais à accès plus 
lent, que l’on appelle « le tas ». Comme il est géré par le framework .NET, on l’appelle 
« le tas managé ». 


age 
| pourcentage 


| 


Pile 10 [55 libre | libre | . 


FIGURE 34.2 -— Les types valeur sont stockés sur la pile 


Si nous avons le code suivant : 


1| public class Voiture 

2 |u € 

3 public int Vitesse { get; set; } 
4 public string Marque { get; set; } 
5 
6 
7 


Voiture maVoiture = new Voiture { Vitesse = 10, Marque = " 
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Peugeot" }; 


nous aurons en mémoire le processus que vous pouvez observer à la figure 34.3. 


maVoiture 


| 


Pile | | libre | libre LS | 


Tes [be [ère Moment [ee 


FIGURE 34.3 — Les types référence ont leur référence allouée sur la pile et leur objet 
alloué sur le tas managé 


À noter que lorsqu'une variable sort de sa portée et qu’elle n’est plus utilisable par 

q q P a P P 
qui que ce soit, alors la case mémoire sur la pile est marquée comme de nouveau libre. 
Voilà grosso modo comment est gérée la mémoire. 


Il manque encore un élément fondamental du mécanisme de gestion mémoire : le 
ramasse-miettes. Nous en avons déjà parlé brièvement, le ramasse-miettes sert à 
libérer la mémoire qui n’est plus utilisée dans le tas managé. Il y aurait de quoi écrire 
un gros tutoriel rien que sur son fonctionnement, aussi nous allons expliquer rapidement 
comment il fonctionne sans trop rentrer dans les détails. 


Le ramasse-miettes est souvent dénommé par sa traduction anglaise, le gar- 
bage collector. 


Pourquoi libérer la mémoire ? On a vu que quand une variable sort de portée, alors la 
case mémoire sur la pile est marquée comme à nouveau libre. C’est très bien avec les 
types valeur qui sont uniquement sur la pile. Mais avec les types référence ? Que donne 
le code suivant lorsque nous quittons la méthode CreerVoiture ? 


1| public class Voiture 

2| € 

3 public int Vitesse { get; set; } 
4 public string Marque { get; set; } 
5| } 

6 

7| static void Main(string[] args) 

8| + 

9 CreerVoiture(); 

10! } 

11 

12] public static void CreerVoiture() 

13| © 
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14 Voiture maVoiture = new Voiture { Vitesse = 10, Marque = " 
Peugeot" }; 
15| } 


Nous avons dit que la mémoire sur la pile redevenait libre. La référence vers l’objet est 
donc cassée. Cependant, la mémoire sur le tas est toujours occupée (voir figure 34.4). 


Pile | libre | libre | libre | : 


Tes [be [ère Moment [ie 


FIGURE 34.4 — L'objet est toujours présent sur le tas, mais n’est plus référencé 


Cela veut dire que notre objet est toujours en mémoire sauf que la variable maVoiture 
n'existe plus et donc, ne référence plus cet objet. Dans un langage comme le C++, 
nous aurions été obligés de vider explicitement la mémoire référencée par la variable. 
En C{, pas besoin! C’est le ramasse-miettes qui le fait pour nous. Encore un truc de 
fainéant ça! 


En fait, c’est plutôt une sécurité qu’un truc de fainéant. C’est la garantie que la mémoire 
utilisée par les objets qui n'existent plus est vidée et, du coup, redevient disponible 
pour de nouveaux objets. Grâce au ramasse-miettes, nous évitons ce que l’on appelle 
les fuites mémoire. 


(1?) C'est super ça mais il passe à quel moment ce ramasse-miettes ? 


La réponse est simple : on ne sait pas quand il passe. En fait, il passe quand il a besoin de 
mémoire ou quand l'application semble ne rien faire. Évidemment, lorsque le ramasse- 
miettes passe, les performances de l’application sont un petit peu pénalisées. C’est 
pour cela que l’algorithme de passage du ramasse-miettes est optimisé pour essayer 
de déranger le moins possible l'application, lors des moments d’inactivité par exemple. 
Par contre, quand il y a besoin de mémoire, il ne se pose pas la question : il passe 
quand même. 


Il y aurait beaucoup de choses à dire encore sur ce mécanisme mais arrêtons-nous 
là. Retenons que le ramasse-miettes est un superbe outil qui nous permet de garantir 
l’intégrité de notre mémoire et qu'il s’efforce de nous embêter le moins possible. Il est 
important de libérer les ressources dont on n’a plus besoin quand c’est possible, par 
exemple en se désabonnant d’un événement ou lorsque nous avons utilisé des ressources 
natives (ce que nous ne verrons pas dans ce livre) ou lors d’accès à des fichiers ou des 
bases de données. 
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Masquer une méthode 


Dans la partie précédente sur la substitution, nous avons vu qu’on pouvait substituer 
une méthode grâce au mot-clé override. Cette méthode devait d’ailleurs s’être déclarée 
comme candidate à cette substitution grâce au mot-clé virtual. Rappelez-vous de ce 
code : 


1| public class Chien 

2| { 

3 public virtual void Aboyer() 

4 { 

5 Console.WriteLine("Wouaf !"); 
6 y 

7|} 

8 

9| public class Caniche : Chien 

10! 

11 public override void Aboyer() 
12 { 
13 Console.WriteLine("Wiiff"); 
14 } 
15| } 


Si nous instancions des chiens et des caniches de cette façon : 


1| static void Main(string[] args) 

2! { 

3 Chien chien = new Chien); 

4 Caniche caniche = new Caniche); 

5 Chien canicheTraiteCommeUnChien = new Caniche); 
6 

7 

8 

9 


chien.Aboyer(); 

caniche.Aboyer(); 

canicheTraiteCommeUnChien.Aboyer (); 
10! } 


Nous aurons : 


En effet, le premier objet est un chien et les deux suivants sont des caniches. Grâce à 
la substitution, nous faisons aboyer notre chien, notre caniche et notre caniche qui se 
fait manipuler comme un chien. 


Il est possible de masquer des méthodes qui ne sont pas forcément marquées comme 
virtuelles grâce au mot-clé new. À ce moment-là, la méthode ne se substitue pas à la 
méthode dérivée mais la masque pour les types dérivés. Voyons cet exemple : 


1 | public class Chien 
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21 { 

3 public void Aboyer() 

4 { 

5 Console.WriteLine("Wouaf !"); 
6 F 

71 

8 

9| public class Caniche : Chien 

10! { 

11 public new void Aboyer() 

12 { 

13 Console.WriteLine("Wiiff"); 
14 } 

15| } 


Remarquons que la méthode Aboyer() de la classe Chien n’est pas marquée virtual 
et que la méthode Aboyer de la classe Caniche est marquée avec le mot-clé new. Il 
ne s’agit pas ici de substitution. Il s’agit juste de deux méthodes qui portent le même 
nom, mais la méthode Aboyer () de la classe Caniche se charge de masquer la méthode 
Aboyer() de la classe mère. Ce qui fait que les précédentes instanciations : 


static void Main(string[] args) 

{ 
Chien chien = new Chien(); 
Caniche caniche = new Caniche); 


1 
2 
3 
4 
5 Chien canicheTraiteCommeUnChien = new Caniche); 
6 
7 chien.Aboyer (); 

8 caniche.Aboyer(); 

9 canicheTraiteCommeUnChien.Aboyer(); 


10| } 


produiront cette fois-ci : 


C’est le dernier cas qui peut surprendre. Rappelez-vous, il ne s’agit pas d’un rempla- 
cement de méthode cette fois-ci, et donc le fait de manipuler le caniche en tant que 
Chien ne permet pas de remplacer le comportement de la méthode Aboyer (). Dans ce 
cas, on appelle bien la méthode Aboyer correspondant au type Chien, ce qui a pour 
effet d'afficher « Wouaf! ». Si nous n’avions pas utilisé le mot-clé new, nous aurions 
eu le même comportement mais le compilateur nous aurait avertis de ceci grâce à un 
avertissement : 


MaPremiereApplication.Caniche.Aboyer ()? masque le membre hérité 


MaPremiereApplication.Chien.Aboyer()’. Utilisez le mot-clé 
new si le masquage est intentionnel. 
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Il nous précise que nous masquons effectivement la méthode de la classe mère et que 
si c’est fait exprès, alors il faut l’indiquer grâce au mot-clé new afin de lui montrer que 
nous savons ce que nous faisons. Notez quand même que dans la majorité des cas, en 
POO, ce que vous voudrez faire, c’est bien substituer la méthode et non la masquer. 


C’est le même principe pour les variables, il est également possible de masquer une 
variable, quoiqu’en général, cela reflète plutôt une erreur dans le code. Si nous faisons : 


1| public class Chien 

2| € 

3 protected int age; 

al} 

5 

6| public class Caniche : Chien 

7| À 

8 private int age; 

9 

10 public Caniche) 

11 { 

12 age = O0; 

13 } 

14 

15 public override string ToString() 
16 { 

17 return "J'ai " + age + " ans"; 
18 } 

19| } 


Alors le compilateur nous indique que la variable age de la classe Caniche masque 
la variable age initialement héritée de la classe Chien. Si c’est intentionnel, il nous 
propose de la marquer avec le mot clé new. Sinon, cela nous permet de constater que 
cette variable est peut-être inutile... (sûrement d’ailleurs !) 


Le mot-clé yield 


Nous avons vu dans le TP sur les génériques comment créer un énumérateur. C’est une 
tâche un peu lourde qui nécessite pas mal de code. Oublions un instant que la classe 
String est énumérable et créons pour l’exemple une classe qui permet d’énumérer une 
chaîne de caractères. Nous aurions pu faire quelque chose comme ça : 


1| public class ChaineEnumerable : IEnumerable <char> 
2| € 

3 private string chaine; 

4 public ChaineEnumerable(string valeur) 

5 

6 chaine = valeur; 

7 1) 

8 

9 public IEnumerator <char> GetEnumerator () 
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10 { 

11 return new ChaineEnumerateur (chaine); 
12 } 

13 

14 IEnumerator IEnumerable.GetEnumerator() 
15 { 

16 return new ChaineEnumerateur (chaine); 
17 } 

18| } 

19 

20| public class ChaineEnumerateur : IEnumerator <char> 
21| € 

22 private string chaine; 

23 private int indice; 

24 

25 public ChaineEnumerateur (string valeur) 
26 { 

27 indice = -1; 

28 chaine = valeur; 

29 } 

30 

31 public char Current 

32 { 

33 get { return chainelindice]l; } 

34 } 

35 

36 public void Dispose() 

37 { 

38 . 

39 

40 object IEnumerator.Current 

al { 

42 get { return Current; } 

43 } 

44 

45 public bool MoveNext() 

46 { 

47 indice++; 

48 return indice < chaine.Length; 

49 } 

50 

51 public void Reset () 

52 { 

5a indice = -1; 

54 } 

55| } 


Vous pouvez copier ce code en utilisant le code web suivant : 


Copier ce code 
Code web : 740399 
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Nous avons une classe ChaineEnumerable qui encapsule une chaîne de caractères de 
type string et une classe ChaineEnumerateur qui permet de parcourir une chaîne et 
de renvoyer à chaque itération un caractère, représenté par le type char ?. 


Rien de bien sorcier, mais un code plutôt lourd. 


Regardons à présent le code suivant : 


1| public class ChaineEnumerable : IEnumerable<char> 
2| { 
3 private string chaine; 
4 public ChaineEnumerable(string valeur) 
5 { 
6 chaine = valeur; 
7 } 
8 
9 public IEnumerator <char> GetEnumerator () 
10 { 
11 for (int i = 0; i < chaine.Length; i++) 
12 { 
13 yield return chainelil]; 
14 } 
15 F 
16 
17 IEnumerator IEnumerable.GetEnumerator () 
18 { 
19 return GetEnumerator(); 
20 } 
21| } 


Il fait la même chose, mais d’une manière bien simplifiée. Nous remarquons l’apparition 
du mot-clé yield. Il permet de créer facilement des énumérateurs. Combiné au mot-clé 
return, il renvoie un élément d’une collection et passe à l’élément suivant. Il peut être 
combiné au mot-clé break également pour permettre d’arrêter l’énumération. Ce qui 
veut dire que nous aurions pu écrire la méthode GetEnumerator() de cette façon : 


1| public IEnumerator <char> GetEnumerator () 
2| € 

3 int i = O0; 

4 while (true) 

5 { 

6 if (i == chaine.Length) 
7 yield break; 

8 yield return chainelil; 
9 i++; 

10 } 

ul} 


Ce qui est plus laïd, je le conçois ! mais qui permet d'illustrer le mot-clé yield break. 


2. Le type char est un alias de la structure System.Char. 
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Le mot-clé yield permet également de renvoyer un IEnumerable (ou sa version géné- 
rique), ainsi un peu bêtement, on pourrait faire : 


1| static void Main(string[] args) 

2| € 

3 foreach (string prenom in ObtenirListeDePrenoms()) 
4 { 

5 Console.WriteLine (prenom); 

6 } 

7|} 

8 

9 | public static IEnumerable<string> ObtenirListeDePrenoms() 
10! 

11 yield return "Nicolas"; 

12 yield return "Jérémie"; 

13 yield return "Delphine"; 

14| } 


Cet exemple va surtout me permettre d'illustrer ce qui se passe exactement grâce au 
mode debug. Mettez un point d’arrêt au début du programme et lancez-le en mode 
debug (voir la figure 34.5). 


=inamespace MaPremiereApplication 
E class Program 
{ 
E static void Main(string[] args) 
=] foreach] (string prenom in ObtenirListeDePrenoms()) 
Console.WriteLine(prenom); 
} 
} 
El public static IEnumerable<string> ObtenirListeDePrenoms() 
{ 
yield return "Nicolas"; 
yield return "Jérémie"; 
yield return "Delphine"; 


sonne 


FIGURE 34.5 — Le programme est en pause sur le point d’arrêt avant l’énumération 


Appuyez sur plusieurs fois pour rentrer dans la méthode ObtenirListeDePrenoms, 
nous voyons l’indicateur se déplacer sur les différentes parties de l’instruction foreach. 
Puis nous arrivons sur la première instruction yield return (voir la figure 34.6). 


Appuyons à nouveau sur et nous pouvons constater que nous sortons de la mé- 
thode et que nous rentrons à nouveau dans le corps de la boucle foreach qui va nous 
afficher le premier prénom. Continuons les jusqu’à repasser sur le mot-clé in 
et rentrer à nouveau dans la méthode ObtenirListeDePrenoms() et nous pouvons 
constater que nous retournons directement au deuxième mot-clé yield, ainsi que vous 
pouvez le voir à la figure 34.7. 


Et ainsi jusqu’à ce qu’il n’y ait plus rien dans la méthode ObtenirListeDePrenoms() 
et que, du coup, le foreach se termine. À la première lecture, ce n’est pas vraiment ce 
comportement que nous aurions attendu... 
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Elnamespace MaPremiereApplication 


{ 
5 class Program 
{ 
A static void Main(string[] args) 
© (string prenom in ObtenirListeDePrenoms()) 
{ 
Console.WriteLine(prenom); 
} 
} 
E public static IEnumerable<string> ObtenirListeDePrenoms() 
{ 
ES 
yield return "Jérémie"; 
yield return "Delphine"; 
} 
F 
100% + 4 


FIGURE 34.6 — Le programme atteint l'instruction yield juste après le foreach 


Enamespace MaPremiereApplication 


{ 


E class Program 


{ 


E static void Main(string[] args) 


{ 
in] (string prenom in ObtenirListeDePrenoms()) 
{ 


} 
} 


E public static IEnumerable<string> ObtenirListeDePrenoms() 


{ 


yield return "Nicolas"; 


Console.WriteLine(prenom); 


yield return "Delphine"; 


} 


+ 


FIGURE 34.7 - Le programme atteint la seconde instruction yield lors de la deuxième 
itération 
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Voilà pour le principe du yield return. 


Notez qu'un yield break nous aurait fait sortir directement de la boucle 
foreach. 


Le mot-clé yield participe à ce qu’on appelle l'exécution différée. On n'évalue la 
méthode ObtenirListeDePrenoms() qu’au moment où on parcoure le résultat avec la 
boucle foreach, ce qui pourrait paraître surprenant. Par exemple, si nous faisons : 


1| public static IEnumerable<string> ObtenirListeDePrenoms() 
2| € 

3 yield return "Nicolas"; 

4 yield return "Jérémie"; 

5 yield return "Delphine"; 

6| } 

7 

8| static void Main(string[]l args) 

g| € 

10 IEnumerable<string> prenoms = ÜUbtenirListeDePrenoms(); 
11 Console.WriteLine("On fait des choses ..."); 

12 foreach (string prenom in prenoms) 

13 { 

14 Console.WriteLine (prenom); 

15 } 

16| } 


et que nous mettons un point d’arrêt sur le Console.WriteLine et un autre dans la 
méthode ObtenirListeDePrenoms(), on s'arrêtera d’abord sur le point d’arrêt de la 
ligne où est situé le Console .WriteLine et nous passerons ensuite dans le point d’arrêt 
positionné dans la méthode ObtenirListeDePrenoms(). En effet, on ne passera dans 
la méthode que lorsqu'on itérera explicitement sur les prénoms grâce au foreach. Nous 
reviendrons sur cette exécution différée un peu plus loin avec le mot-clé yield. 


Le formatage de chaînes, de dates et la culture 


Pour l'instant, lorsque nous avons eu besoin de mettre en forme des chaînes de ca- 
ractères, nous avons utilisé la méthode Console.WriteLine avec en paramètres des 
chaînes concaténées entre elles grâce à l’opérateur +. Par exemple : 


1| int age = 30; 


2| Console.WriteLine("J'ai " + age + " ans"); 


Il est important de savoir qu'il est possible de formater la chaîne un peu plus sim- 
plement et surtout beaucoup plus efficacement. Ceci est possible grâce à la méthode 
string.Format. Le code précédent peut par exemple être remplacé par : 


1 | int age = 30; 
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2| string chaine = string.Format("J'ai {0} ans", age); 
3| Console.WriteLine(chaïine); 


La méthode string.Format accepte en premier paramètre un format de chaîne. Ici, 
nous lui indiquons une chaîne de caractères avec au milieu un caractère spécial : {0}. 
Il permet de dire : « remplace-moi le {0} avec le premier paramètre qui va suivre », 
en l’occurrence la variable age. Si nous avons plusieurs valeurs, il est possible d’uti- 
liser les caractères spéciaux {1}, puis {2}, etc. Chaque nombre représente ici l’indice 
correspondant au paramètre. Par exemple : 


1| double valeuri = 10.5; 

2| double valeur2 = 4.2; 

3| string chaine = string.Format("{0} multiplié par {1} s'écrit 
\"{O} *# {1}\" et donne comme résultat : {2}", valeuri, 
valeur2, valeuri * valeur2); 

4| Console.WriteLine(chaine); 


Ce qui donne : 


10,5 multiplié par 4,2 s’écrit "10,5 * 4,2" et donne comme ré 


sultat : 44,1 


Vous avez pu remarquer qu'il est possible d'utiliser le même paramètre à plusieurs 
endroits. La méthode Console.WriteLine dispose aussi de la capacité de formater des 
chaînes de caractères. Ce qui veut dire que le code précédent peut s’écrire : 


1| double valeuri = 10.5; 

2| double valeur2 = 4.2; 

3| Console.WriteLine("{O0} multiplié par {1} s'écrit \"{O} *x {1}\" 
et donne comme résultat : {2}", valeuri, valeur2, valeuri * 
valeur2); 


Ce qui revient absolument au même. 


Parlons culture désormais. N’ayez pas peur, rien à voir avec des questions pour des 
champions ! La culture en informatique correspond à tout ce qui à trait aux différences 
entre les langues et les paramètres locaux. Par exemple, en France et aux États-Unis, 
les dates s’écrivent différemment. Pour le jour de Noël 2011, voici ce que ça donne : 


— en France : le 25/12/2011: 
— aux États-Unis : the 12/25/2011. 


Le mois et le jour sont donc inversés. De la même façon, il existe des différences lorsque 
nous écrivons les nombres à virgule : 


— en France : 123,45; 
— aux États-Unis : 123.45. 


Un point à la place d’une virgule, subtile différence. On a donc souvent des différences 
entre les langues. Et même plus, on peut avoir des différences entre les langues en 
fonction de l’endroit où elles sont parlées. Citons par exemple le français de France et 
le français du Canada. Il existe donc au sein du système d’exploitation tout une liste de 
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« régions » représentées par un couple de lettres. Par exemple, le français est représenté 
par les lettres « fr ». Et même plus précisément, le français de France est représenté 
par «fr-FR » alors que celui du Canada est représenté par « fr-CA ». C’est ce que 
lon retrouve dans les paramètres de notre système d’exploitation lorsqu'on va (sous 
Windows 7) dans le panneau de configuration, Horloge langue et région, comme 
vous pouvez le voir à la figure 34.8. 


[ (NIET 
7 
( XX) = ÎE3 » Panneau de configuration » "4 | Rech ? 
Ajuster les paramètres de l'ordinateur Afficher par: Catégorie * 
M Système et sécurité Comptes et protection des utilisateurs 
U Consulter l'état de votre ordinateur # Ajouter ou supprimer des comptes d'utilisateurs 
Sauvegarder l'ordinateur # Configurer le contrôle parental pour un utilisateur 
Rechercher et résoudre des problèmes 
É Apparence 
Réseau et Internet y Modifier l'arrière-plan du Bureau 
M Se connecter à Internet Modifier la résolution de l'écran 
“æ. Afficher l'état et la gestion du réseau 
Choisir les options de groupe résidentiel et de Horloge, langue et région 
persge 1e Modifier les claviers ou les autres méthodes d'entrée 
{ ee € .— À Ortions d'ergonomie 
el sales FES SES Sp { y Laisser Windows suggérer les paramètres 
RET Le is sans Optimiser l'affichage 
Se connecter à un projecteur 
Ajuster les paramètres de mobilité communément 
utilisés: 
= Programmes 
(i | Désinstaller un programme 
D 2 


FIGURE 34.8 — Choisir Horloge, langue et région dans le panneau de configuration 


Si vous allez dans Modifier le format de la date, de l’heure ou des nombres 
(voir figure 34.9), vous pouvez adapter le format à celui que vous préférez (voir la figure 
34.10). 


Bref, ce choix, nous le retrouvons dans notre application si nous récupérons la culture 
courante : 


1| Console.WriteLine (System.Threading.Thread.CurrentThread. 


CurrentCulture); 


Avec ceci, nous aurons : 


Ce qui est intéressant, c’est que le formatage des chiffres ou des dates change en fonction 
de la culture. Ainsi, si nous utilisons le français de France, pour le code suivant : 


1| public static void Main(string[]l args) 
2| € 
3 Affiche (); 
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LE) 
lee) @ » Panneau de configuration + Horloge, langue et région + [6 Î] Rechercher p 


Page d'accueil du panneau de 


des Date et heure 
Définir l'heure et la date | Modifier le fuseau horaire 

Système et sécurité Ajouter des horloges pour des fuseaux horaires variés | Ajouter le gadget Horloge au Bureau 

Réseau et Internet Région et langue 

Matériel et audio Modifier l'emplacement À Modifier le format de la date, de l'heure ou des nombres 
Modifier les claviers ou les teen 

Programmes 

Comptes et protection des 

utilisateurs 

Apparence 

+ Horloge, langue et région 
Options d'ergonomie 


FIGURE 34.9 — On peut modifier ici le format de la date 


Emplacement | Claviers et langues | Administration 
Format : 


Filipino (Philippines) 
Finnois (Finlande) 
Français (Belgique) 


pans cars 


Français (Luxembourg) 

Français (Monaco) 

Français (Suisse) 

Frison (Pays-Bas) 

Gaélique écossais (Royaume-Uni) 

Galicien (Galicien) 

Gallois (Royaume-Uni) 

Géorgien (Géorgie) 

Goudjrati (Inde) 

Grec (Grèce) 

Groenlandais (Groenland) 

Haoussa (latin, Nigeria) 

Haut-sorabe (Allemagne) 

Hébreu (Israël) 

Hindi (Inde) 

Hongrois (Hongrie) 

lakoute (Russie) 

Igbo (Nigeria) 

Indonésien (Indonésie) 

L_!Inuktitut (latin, Canada) 
Inuktitut (syllabique, Canada) 
Irlandais (Irlande) 

> Islandais (Islande) 

Italien (Italie) 


FIGURE 34.10 — Sélection de la langue pour le format 
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al } 

5 

6| public static void Affiche) 

7|{ 

8 Console .WriteLine (System.Threading.Thread.CurrentThread. 
CurrentCulture); 

9 decimal dec = 5.5M; 

10 double dou = 4.8; 

11 DateTime date = new DateTime(2011, 12, 25); 

12 Console.WriteLine("Décimal : {0}", dec); 

13 Console.WriteLine ("Double : {0}", dou); 

14 Console.WriteLine("Date : {0}", date); 

15| } 


nous aurons : 


fr-FR 
Décimal : 5,5 


Double : 4,8 
Date : 25/12/2011 00:00:00 


Il est possible de changer la culture courante de notre application, si par exemple nous 
souhaitons qu’elle soit multiculture, en utilisant le code suivant : 


1| System.Threading.Thread.CurrentThread.CurrentCulture = new 
Culturelnfo("en-US"); 


Ici par exemple, j'indique à mon application que je souhaite travailler avec la culture 
des Etats-Unis. Ainsi, le code précédent, en changeant juste la culture : 


1| public static void Main(string[]l args) 


2| { 

3 System.Threading.Thread.CurrentThread.CurrentCulture = new 
Culturelnfo("en-US"); 

4 Affiche (); 

51} 

6 

7| public static void Affiche() 

8| { 

9 Console.WriteLine(System.Threading.Thread.CurrentThread. 
CurrentCulture); 

10 decimal dec = 5.5M; 

11 double dou = 4.8; 

12 DateTime date = new DateTime(2011, 12, 25); 

13 Console.WriteLine("Décimal : {0}", dec); 

14 Console.WriteLine ("Double : {0}", dou); 

15 Console.WriteLine ("Date : {0}", date); 

16| } 


produira un résultat légèrement différent : 
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en-US 
Décimal : 5.5 


Double : 4.8 
Date-…12/25/2011. 12:00:00 AM 


Il s’agit exactement de la même information, mais formatée différemment. 


À noter qu’il existe encore une autre information de culture, à savoir la langue du 
système d'exploitation que l’on retrouve dans la culture de l’interface graphique, dans 
la propriété : 


1| Console.WriteLine(System.Threading.Thread.CurrentThread. 
CurrentUIlCulture); 


Il s’agit de la CurrentUlCulture, à ne pas confondre avec la CurrentCulture. Corres- 
pondant à la langue du système d’exploitation, nous nous en servirons plus volontiers 
pour rendre une application multilingue. 


Pour finir sur le formatage des chaînes, il est possible d’utiliser des caractères spéciaux 
enrichis pour changer le formatage des types numériques ou de la date. Par exemple, 
il est possible d'afficher la date sous une forme différente avec : 


1| DateTime date = new DateTime(2011, 12, 25); 
2| Console.WriteLine("La date est {0}", date.ToString("dad-MM-yyyy" 
12 


Ici, « dd » sert à afficher le jour, « MM » le mois et & yyyy » l’année sur quatre chiffres, 
ce qui donne : 


La date est 25-12-2011 


Il y a beaucoup d'options différentes que vous pouvez retrouver dans la documentation 
Microsoft, via le code web suivant : 


DateTime 
Code web : 781180 


Le fonctionnement est similaire pour les numériques ; ce qui nous permet par exemple 
d'afficher un nombre de différentes manières. 


Ici par exemple, j'utilise la notation scientifique : 


1 
2 


double valeur = 123.45; 
Console.WriteLine ("Format scientifique : {0:e}", valeur); 


Ce qui donne : 


Format scientifique : 1,234500e+002 


Pour les nombres, il existe différents formatages que l’on peut retrouver dans le tableau 
ci-après. 
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Format Description Remarque 

{0 :c} Monnaie Attention, la console affiche mal le carac- 
tère € 

{0 :d} Décimal Ne fonctionne qu'avec les types sans vir- 
gules 

{0 :e} Notation scientifique 

{0 :f} Virgule flottante 

{0 :g} Général Valeur par défaut 

{0 :n} Nombre avec formatage 

pour les milliers 

{0 :r} Aller-retour Permet de garantir que la valeur numérique 
convertie en chaîne pourra être convertie à 
nouveau en valeur numérique identique 

{0 :x} Hexadécimal Ne fonctionne qu'avec les types sans vir- 
gules 


On peut également avoir du formatage grâce à la méthode ToString(). Il suffit de 
passer le format en paramètres, comme ici, où cela me permet d’afficher uniquement le 
nombre de chiffres après la virgule qui m'’intéressent ou un nombre des chiffres signifi- 
catifs : 


1| double valeur = 10.0 / 3; 
2| Console.WriteLine("Avec 3 chiffres significatifs et 3 chiffres 
après la virgule : {0}", valeur.ToString("000.###")); 


Qui donne : 


Avec 3 chiffres significatifs et 3 chiffres après la virgule : 
003,333 


Mettre le format dans la méthode ToString() fonctionne aussi pour la date : 


1 
2 


DateTime date = new DateTime(2011, 12, 25); 
Console.WriteLine(date.ToString("MMMM")); 


Ici, je n'affiche que le nom du mois : 


Voilà pour le formatage des chaînes de caractères. Lister tous les formatages possibles 
serait une tâche bien ennuyeuse. Vous aurez l’occasion de rencontrer d’autres forma- 
tages qui permettent de faire un peu tout et n’importe quoi. Vous pourrez retrouver la 
plupart des formatages à partir de ce code web : 


Formatages ) 


Code web : 500571 
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Les attributs 


Dans la liste des choses qui vont vous servir, on peut ajouter les attributs. Ils sont 
utilisés pour fournir des informations complémentaires sur une méthode, une classe, des 
variables, etc. Ils vont nous servir régulièrement dans le framework .NET et nous aurons 
même la possibilité de créer nos propres attributs. Un attribut est déclaré entre crochets 
avec le nom de sa classe. Il peut se mettre au-dessus d’une classe, d’une méthode, d’une 
propriété, etc. Par exemple, dans le code ci-dessous, je positionne l’attribut Obsolete 
au-dessus de la déclaration d’une méthode : 


1| [Obsolete("Utilisez plutôt la méthode ToString() pour avoir une 
représentation de l'objet")] 


2| public void Affiche) 
3| À 

4 // 

5| } 


Ici, j'utilise un attribut existant du framework .NET, l’attribut Obsolete qui permet 
d'indiquer qu’une méthode est obsolète et qu’il vaudrait mieux ne pas l'utiliser. En 
général, il est possible de fournir une information permettant de dire pourquoi la mé- 
thode est obsolète et par quoi il faut la remplacer. Aïnsi, si nous tentons d'utiliser cette 
méthode : 


1] public class Program 
2| € 
3 static void Main(string[]l args) 
4 { 
5 new Program().Affiche(); 
6 } 
7 
8 [Obsolete("Utilisez plutôt la méthode ToString() pour avoir 
une représentation de l'objet")] 
9 public void Affiche) 
10 { 
il // 
12 } 
13| } 


Le compilateur nous affichera l’avertissement suivant : 


MaPremiereApplication.Program.Affiche()? est obsolète : ? 


Utilisez plutôt la méthode ToString() pour avoir une repré 
sentation de l’objet”? 


Cet attribut est un exemple, il en existe beaucoup dans le framework .NET et vous 
aurez l’occasion d’en voir d’autres dans les chapitres suivants. Remarquons que la classe 
s'appelle ObsoleteAttribute et peut s’utiliser soit suffixée par Attribute, soit sans, 
et dans ce cas, ce suffixe est ajouté automatiquement. 


Grâce à l’héritage, nous allons nous aussi pouvoir définir nos propres attributs. Il suffit 
pour cela de dériver de la classe de base Attribute. Par exemple, dans sa forme la plus 
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simple : 

1| public class DescriptionClasseAttribute : Attribute 
2| € 

3[} 


Nous pourrons alors utiliser notre attribut sur une classe : 


[DescriptionClasse] 
public class Chien 
{ 
} 


Æ © ND 


Super, même si ça ne nous sert encore à rien! 


Problème, j’ai appelé mon attribut DescriptionClasse et je peux quand même l’uti- 
liser sur une méthode : 


1| [DescriptionClasse] 

2| public class Chien 

3| + 

4 [DescriptionClassel] 
5 public void Aboyer() 
6 { 

7 } 

8|} 


Heureusement, il est possible d'indiquer des restrictions sur les attributs en indiquant 
par exemple qu’ils ne sont valables que pour les classes. Cela se fait avec... un attribut! 


1| [AttributeUsage(AttributeTargets.Class, AllowMultiple = true)] 


2| public class DescriptionClasseAttribute : Attribute 
3| + 
al} 


Ici par exemple, j’indique que mon attribut n’est valable que sur les classes et qu’il 
est possible de le mettre plusieurs fois sur la classe. Le code précédent où l’attribut 
est positionné sur une méthode ne compilera donc plus, avec l’erreur de compilation 
suivante : 


L’attribut ?DescriptionClasse? n’est pas valide dans ce type de 


déclaration. Il n’est valide que dans les déclarations ?class 
] 


Ce qui nous convient parfaitement ! 


Il existe d'autres attributs qui permettent par exemple de n'autoriser des 
attributs que sur les méthodes. 


Il est intéressant de pouvoir fournir des informations complémentaires à notre attribut. 
Pour cela, on peut rajouter des propriétés, ou avoir une surcharge complémentaire du 
constructeur, comme on l’a fait pour le message de l’attribut Obsolete : 
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1| [AttributeUsage (AttributeTargets.Class, AllowMultiple = true)] 
2| public class DescriptionClasseAttribute : Attribute 

3| 

4 public string Description { get; set; } 

5 

6 public DescriptionClasseAttribute() 

7 { 

8 } 

9 

10 public DescriptionClasseAttribute(string description) 
11 { 

12 Description = description; 

13 } 

14/ } 


Ce qui nous permettra d'utiliser notre attribut de ces deux façons : 


1| [DescriptionClasse (Description = "Cette classe correspond à un 
chien")] 

2| [DescriptionClasse("Elle dérive de la classe Animal")] 

3| public class Chien : Animal 

al À 

5| } 


C’est très bien ces attributs, mais nous ne sommes pas encore capables de les exploiter. 
Voyons maintenant comment le faire. 


La réflexion 


La réflexion en C# ne donne pas mal à la tête, quoique...! C’est une façon de faire 
de l’introspection sur nos types de manière à obtenir des informations sur eux, c’est- 
à-dire sur les méthodes, les propriétés, etc. La réflexion permet également de charger 
dynamiquement des types et de faire de l’introspection sur les assemblys. C’est un 
mécanisme assez complexe et plutôt coûteux en terme de temps. Le point de départ 
est la classe Type, qui permet de décrire n’importe quel type. Le type d’une classe peut 
être obtenu grâce au mot-clé typeof : 


1 | Type type = typeof(string); 
Cet objet Type nous permet d’obtenir plein d'informations sur nos classes au moment 


de l’exécution. Par exemple si le type est une classe, c’est-à-dire ni un type valeur, ni 
une interface, alors le code suivant : 


1| Type type = typeof (string); 
2| Console.WriteLine (type.IsClass); 


nous renverra true. En effet, la propriété IsClass permet d’indiquer si le type est une 
classe. Nous pouvons aussi obtenir le nom de méthodes grâce à GetMethods() : 
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1| Type type = typeof (string); 

2| foreach (MethodInfo infos in type.GetMethods()) 
3| À 

4 Console.WriteLine(infos.Name); 

5|} 


Je vous renvoie à la figure 34.11 pour le résultat dans la console. 


Fr 


CAWindows\system32\cmd.exe [ cs) 


Appuvez sur une touche pour continuer... 


FIGURE 34.11 — Affichage du nom des méthodes 


Pas très exploitable comme résultat, mais bon, c’est pour l’exemple! Il y a encore 
pleins d’infos dans cet objet Type, qui permettent d'obtenir ce que nous voulons. Cela 
est possible grâce aux métadonnées qui sont des informations de description des types. 


Vous me voyez venir... eh oui, nous allons pouvoir obtenir nos attributs. Pour connaître 

la liste des attributs d’un type, on peut utiliser la méthode statique 

Attribute.GetCustomAttributes(), en lui passant en paramètre le System.Type qui 

est hérité de la classe abstraite Memberlnfo. Si l’on souhaite récupérer un attribut 

particulier, on peut le passer en paramètre de la méthode : 

1| Attribute[] lesAttributs = Attribute.GetCustomAttributes(type, 
typeof (DescriptionClasseAttribute)); 


Ceci va nous permettre de récupérer les attributs de notre classe et en l’occurrence, 
d'obtenir leurs descriptions. C’est ce que permet le code suivant : 


1| public class Program 


2| { 

3 static void Main(string[] args) 
4 { 

5 new DemoAttributs().Demo(); 
6 } 

7 

8| } 
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10| public class DemoAttributs 

11 À 

12 public void Demo() 

13 { 

14 Animal animal = new Animal); 

15 Chien chien = new Chien); 

16 

1 VoirDescription(animal); 

18 VoirDescription(chien); 

19 d 

20 

21 public void VoirDescription<T>(T obj) 

22 { 

23 Type type = typeof(T); 

24 if (ltype.lsClass) 

25 return; 

26 Attribute[] lesAttributs = Attribute. 
GetCustomAttributes(type, typeof( 
DescriptionClasseAttribute)); 

27 if (lesAttributs.Length == 0) 

28 Console.WriteLine("Pas de description pour la 

classe " + type.Name + "\n"); 

29 else 

30 { 

31 Console.WriteLine ("Description pour la classe " + 

type.Name); 

32 foreach (DescriptionClasseAttribute attribut in 

lesAttributs) 

33 Le 

34 Console.WriteLine("\t" + attribut.Description); 

35 } 

36 } 

37 } 

38| } 


Nous commençons par instancier un objet animal et un objet chien. Puis nous de- 
mandons leurs descriptions. La méthode générique VoirDescription() s'occupe dans 
un premier temps d'obtenir le type de la classe grâce à typeof. On s’octroie une vérifi- 
cation permettant de nous assurer que le type est bien une classe. Ceci permet d'éviter 
d’aller plus loin car de toute façon, notre attribut ne s'applique qu’aux classes. Puis 
nous utilisons la méthode GetCustomAttributes pour obtenir les attributs de type 


DescriptionClasseAttribute. S'il y en a, alors on les affiche. 


Ce qui donne : 


Pas de description pour la classe Animal 


Description pour la classe Chien 
Elle dérive de la classe Animal 
Cette classe correspond à un chien 
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Voilà pour les attributs, vous aurez l’occasion de rencontrer des attributs du framework 
.NET un peu plus loin dans ce tutoriel, mais globalement, il est assez rare d’avoir à 
créer soi-même ses attributs. 


En ce qui concerne la réflexion, il faut bien comprendre que le sujet est beaucoup plus 
vaste que ça. Nous avons cependant vu ici un aperçu de ce que permet de faire la 
réflexion, en illustrant le mécanisme d’introspection sur les attributs. 


En résumé 


— Ilest possible d'empêcher une classe d’être dérivée grâce au mot-clé sealed. 

Le garbage collector est le mécanisme permettant de libérer la mémoire allouée sur 

le tas managé qui n’est plus référencée dans une application. 

— Le mot-clé yield permet de créer des énumérateurs facilement et participe au mé- 
canisme d’exécution différée. 

— Le framework .NET gère les différents formats des chaînes grâce à la culture de 
l’application. 

— La réflexion est un mécanisme d’introspection sur les types permettant d’obtenir des 
informations sur ces derniers lors de l’exécution. 
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La configuration d'une application 


Difficulté : 53 


ans le cycle de vie d'une application, il est fréquent que de petites choses changent. 
Imaginons que mon application doive se connecter à un serveur FTP pour sauvegarder 
les données sur lesquelles j'ai travaillé. Pendant mes tests, effecutés en local, mon 
serveur FTP pourrait avoir l'adresse ftp://localhost avec un login et un mot de passe 
« test ». Évidemment, lors de l'utilisation réelle de mon application, l'adresse changera, 
et le login et le mot de passe varieront en fonction de l'utilisateur. Il faut être capable de 
changer facilement ces paramètres sans avoir à modifier le code ni à recompiler l'application. 


C'est là qu'interviennent les fichiers de configuration : ils permettent de stocker toutes ces 
petites choses qui servent à faire varier notre application. Pour les clients lourds, comme 
nos applications console, il s'agit du fichier app.config : un fichier XML qui contient 
la configuration de notre application. Il stocke des chaînes de connexion à une base de 
données, une url de service web, des préférences utilisateurs, etc. Bref, tout ce qui va 
permettre de configurer notre application! 
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Rappel sur les fichiers XML 


Vous ne connaissez pas les fichiers XML ? Si vous voulez en savoir plus, n’hésitez pas 
à faire un petit tour sur internet, c’est un format très utilisé dans l’informatique. Pour 
faire court, le XML est un langage de balise, un peu comme le HTML, où l’on décrit 
de l’information. Les balises sont des valeurs entourées de < et > qui décrivent la 
sémantique de la donnée. Par exemple : 


1 | <prenom>Nicolas</prenom> 


La balise <prenom> est ce qu’on appelle une balise ouvrante, cela signifie que ce qui se 
trouve après (en l’occurrence la chaîne « Nicolas ») fait partie de cette balise jusqu’à 
ce que l’on rencontre la balise fermante </prenom> qui est comme la balise ouvrante à 
l'exception du / précédant le nom de la balise. Le XML est un fichier facile à lire par 
nous autres humains. On en déduit assez facilement que le fichier contient la chaîne 
« Nicolas » et qu’il s’agit sémantiquement d’un prénom. Une balise peut contenir des 
attributs permettant de donner des informations sur la donnée. Les attributs sont 
entourés de guillemets : "" et font partie de la balise. Par exemple : 


1 | <client nom="Nicolas" age="30"></client> 


Ici, la balise client possède un attribut nom ayant la valeur « Nicolas » et un attribut 
age ayant la valeur « 30 ». Encore une fois, c’est très facile à lire pour un humain. 
Il est possible que la balise n’ait pas de valeur, comme c’est le cas dans l’exemple ci- 
dessus. On peut dans ce cas-là remplacer la balise ouvrante et la balise fermante par 
cet équivalent : 


1] <client nom="Nicolas" age="30"/> 


Enfin, et nous allons terminer notre aperçu rapide du XML avec ce dernier point, il 
est important de noter que le XML peut imbriquer ses balises et qu’il peut ne posséder 
qu’un seul élément racine, ce qui nous permet d’avoir une hiérarchie de données. Par 
exemple nous pourrons avoir : 


<listesDesClient> 
<client type="Particulier"> 
<nom>Nicolas</nom> 
<age>30</age> 
</client> 
<client type="Professionel"> 
<nom>Jérémie</nom> 
<age>40</age> 
</client> 
</listesDesClient> 


© ® I OO Où À À NN H 


= 
[e} 


On voit tout de suite que le fichier décrit une liste de deux clients. Nous en avons un 
qui est un particulier, qui s'appelle Nicolas et qui a 30 ans alors que l’autre est un 
professionnel, prénommé Jérémie et qui a 40 ans. 
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Créer le fichier de configuration 


Pourquoi utiliser un fichier de configuration ? 

— Pour éviter de mettre des valeurs en dur dans le code. Imaginons que nous utilisions 
une url de service web dans notre application, si l’url change, on aimerait ne pas 
avoir à recompiler le code. 

— Pour éviter d'utiliser la base de registre comme cela a beaucoup été fait dans les 
premières versions de Windows et qui oblige à donner les droits de modification sur 
celle-ci. 

Ces fichiers permettent d’avoir des informations de configuration, potentiellement ty- 

pées. De plus, le framework .NET dispose de méthodes pour y accéder facilement. 

L'intérêt d'utiliser un fichier XML plutôt qu’un fichier binaire est que ce fichier est 

lisible et compréhensible facilement. On peut également le modifier à la main sans 

un système évolué permettant de faire des modifications. Voilà plein de raisons pour 
lesquelles on va utiliser ces fichiers. 


Pour ajouter un fichier de configuration : faisons un clic droit sur le projet : Ajouter 
> Nouvel élément, comme l’illustre la figure 35.1. 


+ | 3% Main(string[] args) E 


77 “remiereApplication 
5 Géné 
ES] Générer rs 
Régénérer SE 
là Publier. ogram.cs 


LEA Nouvel élément. Ctd+MajrA || Ajouter » 
E] Élément existant... Maj+Alt+A Ajouter une référence... 
Ca Nouveau dossier Ajouter une référence de service... 
1 Formulaire Windows. Définir comme projet de démarrage 
‘Contrôle utilisateur. Déboguer » 
Ÿg Classe. Maj+Ak+C |% Couper Ctri+X 
M Coller Ctri+V 
X Supprimer Suppr 
Renommer 
3 Propriétés Alt+Entrée 


FIGURE 35.1 — Ajout du fichier de configuration 


Choisissons le modèle de fichier Fichier de configuration de l’application (voir 
figure 35.2). 


Conservez son nom et validez sa création. Ce fichier est presque vide : 


1] <?xml version="1.0" encoding="utf-8" ?> 
2| <configuration> 
3| </configuration> 


Il possède sur sa première ligne un marqueur permettant d'identifier le fichier comme 
étant un fichier XML et une balise ouvrante configuration, suivie de sa balise fer- 
mante. C’est la structure de base du fichier de configuration. Nous mettrons nos sections 
de configuration à l’intérieur de cette balise. Nous retrouvons notre fichier de configu- 
ration dans notre projet et nous pouvons le voir dans l’explorateur de documents, ainsi 
que l’illustre la figure 35.3. 
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Éléments Visual C# 


Modèles en ligne 


Nom: 


App.config 


ict = 
|| Trier par: [Par défaut AE | Rechercher Modèles installés »| 
a à : Élé Visual C# 
tu Base de données locale Éléments Visual C# 
Fichier pour stocker la configuration de 
: l'application et les valeurs des paramètres 
Boîte de dialogue À propos de Éléments Visual C# 
«2 classes LINQ to SQL Éléments Visual C# 
EF DataSet Éléments Visual C# 
a] Fichier de code Éléments Visual C# 


Fichier de paramètres Éléments Visual C# 
a Fichier de ressources Éléments Visual C# 
2 Fichier d'informations de l'assembly Éléments Visual C# 
la) Fichier manifeste d'application Éléments Visual C# 
É Fichier texte Éléments Visual C# + 


FIGURE 35.2 — Choix et nommage du fichier de configuration 


Explorateur de solutions 


#22 
| Solution 'MaPremiereApplication' (1 projet) 
4 KL] MaPremi ir 

b (5 Properties 


ÉS 


FIGURE 35.3 — Le fichier de configuration est présent dans le projet 


LECTURE SIMPLE DANS LA SECTION DE CONFIGURATION PRÉDÉFINIE : 
APPSETTINGS 


Compilons notre application et rendons-nous dans le répertoire où le fichier exécutable 
est généré. Nous y trouvons également un fichier qui porte le même nom que notre exé- 
cutable et dont l’extension est .exe.config. C’est bien notre fichier de configuration. 
Visual C# Express le déploie au même endroit que notre application et lui donne un 
nom qui va lui permettre d’être exploité par notre application. Pour être utilisables, 
ces fichiers doivent se situer dans le même répertoire que l’exécutable. 


Lecture simple dans la section de configuration prédé- 
finie : AppSettings 


Avoir un fichier de configuration, c’est bien. Mais il faut savoir lire à l’intérieur si nous 
souhaitons l’exploiter. Il existe plusieurs façons d’indiquer des valeurs de configuration, 
nous les découvrirons tout au long de ce chapitre. La plus simple est d'utiliser la section 
AppSettings. C’est une section où l’on peut mettre, comme son nom le suggère aux 
anglophones, les propriétés de l’application. Pour cela, on utilisera un système de clé ou 
valeur pour stocker les informations. Par exemple, modifiez le fichier de configuration 
pour avoir : 


1| <configuration> 

2 <appSettings> 

3 <add key="prenom" value="nicolas"/> 
4 <add key="age" value="30"/> 

5 </appSettings> 

6| </configuration> 


En voyant ce fichier de configuration, un être humain comprendra assez facilement 
qu'il y a deux paramètres de configuration. Un premier qui est le prénom et qui va 
valoir Nicolas. Un deuxième qui est l’âge et qui sera de 30. Le savoir en tant qu'être 
humain, c’est bien. Pouvoir y accéder dans notre programme informatique, c’est mieux! 
Voyons comment faire. La première chose à faire est de référencer, si ce n’est déjà fait, 
l’assembly qui contient toutes les classes permettant de gérer la configuration. Elle 
s’appelle (sans surprise) System.Configuration (voir figure 35.4). 


Afin d'accéder au fichier de configuration, nous allons devoir utiliser la classe sta- 
tique ConfigurationManager. Pour accéder aux informations contenues dans la section 
AppSettings, nous utiliserons sa propriété AppSettings. Et nous pourrons accéder aux 


éléments de la configuration en utilisant l'opérateur d’indexation : [[. Ce qui donne : 
1| string prenom = ConfigurationManager.AppSettings['prenom'"]; 

2| string age = ConfigurationManager.AppSettings["age"]; 

3 

4| Console.WriteLine("Prénom : " + prenom + " / Age : " + age); 


On peut également utiliser des index numériques pour y accéder, mais je trouve que ça 
manque vraiment de clarté : 


1| string prenom = ConfigurationManager.AppSettings[O]; 
2| string age = ConfigurationManager.AppSettings[1]; 


401 


CHAPITRE 35. LA CONFIGURATION D'UNE APPLICATION 


NET |COM | Projets | Parcourir | Récent 
Filtré pour : .NET Framework 4 Client Profile 
Nom du composant Version Runtime Chemin d'accès a 
System.Addin 40.00 v4.0.30319 C:\Program Files\Referen.. 
System.ComponentModel….  4.0.0.0 v4.0.30319 C:\Program Files\Referen… __ 
System.ComponentModel.… 4.0.0.0 v4.0.30319 C:\Program Files\Referen... E] 
em.Configuration 4.0.0.0 v4.0.30319  C:\P: Files\Referen.… 
System.Configuration.install 4.0.0.0 v4.0.30319 C:\Program Files\Referen.. 
System.Core 40.00 v4.0.30319 C:\Program Files\Referen.. 
System.Data.DataSetExtens. 4.0.0.0 v4.0.30319 C:\Program Files\Referen... 
System.Data 40.00 v4.0.30319 C:\Program Files\Referen.. 
System.Data.Entity 40.00 v4.0.30319 C:\Program Files\Referen.. 
System.Data.Linq 40.00 v4.0.30319 C:\Program Files\Referen... 
System.Data.Services.Client  4.0.0.0 v4.0.30319 C:\Program Files\Referen… + 
« | TT | p 


FIGURE 35.4 — Ajout de la référence à l’assembly System.Configuration 


3 


4| Console.WriteLine ("Prénom " + prenom + " / Age " + age); 
De plus, cela oblige à connaître l’ordre dans lequel les clés ont été mises dans le fichier. 


Il est possible aussi de boucler sur toutes les valeurs contenues dans la section : 


1| foreach (string cle in ConfigurationManager.AppSettings) 

2| € 

3 Console.WriteLine("Clé : " + cle + " / Valeur : " + 
ConfigurationManager.AppSettings[clel]); 

al} 


La propriété AppSettings est en fait du type NameValueCollection. Il s’agit d’une 


collection d'éléments de type chaîne de caractères qui sont accessibles à partir d’une 
clé de type chaîne de caractères également. Une valeur est donc associée à une clé. 


Pour plus d’informations sur le type NameValueCollection, je vous invite à consulter 
la documentation en ligne, via le code web suivant : 


NameValueCollection 
Code web : 357840 


À noter que la casse de la clé n’est pas importante pour accéder à la valeur associée à 
la clé. Le code suivant : 


1 | string prenom = ConfigurationManager.AppSettings ['PRENOM"]; 


renverra bien notre valeur de configuration. Cependant, si la valeur n’existe pas, nous 
obtiendrons la valeur null dans la chaîne de caractères. Il est important de remarquer 
que nous récupérons des chaînes et que nous aurons besoin potentiellement de faire une 
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conversion pour manipuler l’âge en tant qu’entier. Rien ne vous empêche d'écrire une 
petite méthode d’extension maintenant que vous savez faire : 


1| public static class ConfigurationManagerExtensions 
2| € 
3 public static int UbtenirValeurEntiere(this 
NameValueCollection appSettings, string cle) 
À 
string valeur = appSettings[cle]; 
return Convert.Tolnt32(valeur) ; 


œ ] o où 


Que nous pourrons utiliser avec : 


1| int age = ConfigurationManager .AppSettings.0btenirValeurEntiere 
("age"); 


Lecture des chaînes de connexion à la base de données 


Les chaînes de connexion représentent un type de configuration particulier. Elles vont 
servir pour les applications ayant besoin de se connecter à une base de données. On va 
y stocker tout ce dont on a besoin, comme le nom du serveur ou les identifiants pour s’y 
connecter... Nous y reviendrons en détail plus tard, mais regardons la configuration 
suivante : 


1| <configuration> 


Le] 


<connectionStrings?> 


3 <add name="MaConnection" providerName="System.Data. 
SqlClient" 
4 connectionString="Data Source=.\SQLEXPRESS; Initial 
Catalog=Basel; Integrated Security=true"/> 
5 <add name="MaConnection2" providerName="System.Data. 
SqlClient" 
6 connectionString="Data Source=.\SQLEXPRESS; Initial 
Catalog=Base2; Integrated Security=true"/> 
7 </connectionStrings> 


8| </configuration> 


Nous définissons ici deux chaînes de connexion qui permettent de se connecter en 
authentification Windows (Integrated Security=true) sur un serveur hébergé en 
local (.\SQLEXPRESS) dont le nom est SQLEXPRESS, qui utilisent SQL SERVER 
(providerName="System.Data.SqlClient"), qui pointent sur les bases de données 
Base1 et Base2, identifiées chacune par les noms MaConnection et MaConnection2. 
Pour obtenir les informations de configuration individuellement, il faudra utiliser la 
propriété ConnectionStrings de la classe ConfigurationManager en y accédant par 
leurs noms : 
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1] ConnectionStringSettings chaineConnexion = ConfigurationManager 
.ConnectionStrings["MaConnection"l]; 
Console.WriteLine (chaineConnexion.Name); 
Console.WriteLine (chaineConnexion.ConnectionString); 
Console.WriteLine (chaineConnexion.ProviderName); 


Nous pourrons toutes les obtenir avec une boucle foreach : 


1| foreach (ConnectionStringSettings valeur in 
ConfigurationManager.ConnectionStrings) 


{ 


© ND 


Console.WriteLine(valeur.ConnectionString); 


al } 


Nous utiliserons les chaînes de connexion dans le chapitre sur l’accès aux données. 


Créer sa section de configuration depuis un type pré- 
défini 


Il est possible de créer sa propre section de configuration à partir d’un type prédéfini. 
Par exemple pour créer une section du même genre que la section appSettings, qui 
utilise une paire clé / valeur, on peut utiliser un DictionarySectionHandler. Il existe 
plusieurs types prédéfinis, que nous allons étudier ci-dessous. 


Le type DictionarySectionhandler 


DictionarySectionhandler est une classe qui fournit les informations de configuration 
des paires clé / valeur d’une section de configuration. 


i is si i 'est un système de clé / valeur, c'est comme 
Oui mais si tu nous dis que c 
pour les AppSettings que nous avons vus. Pourquoi utiliser une section 
particulière DictionarySectionhandler alors ? 


L'intérêt de pouvoir faire des sections particulières est d'organiser sémantiquement son 
fichier de configuration, pour découper logiquement son fichier au lieu d’avoir tout dans 
la même section. Regardons cette configuration : 


1| <configuration?> 
2 <configSections?> 
3 <section name="InformationsUtilisateur" type="System. 
Configuration.DictionarySectionHandler" /> 

</configSections> 
<InformationsUtilisateur> 

<add key="login" value="nico" /> 

<add key="motdepasse" value="12345" /> 

<add key="age" value="30" /> 
</InformationsUtilisateur> 


© © J OO Où À 
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10 | </configuration> 


La première chose à voir est que nous indiquons à notre application que nous défi- 
nissons une section de configuration du type DictionarySectionHandler et qui va 
s'appeler InformationsUtilisateur. Cela permet ensuite de définir notre propre sec- 
tion InformationsUtilisateur, qui ressemble beaucoup à la section AppSettings 
sauf qu'ici, on se rend tout de suite compte qu'il s’agit d'informations utilisateurs. 
Pour accéder à notre section depuis notre programme, nous devrons utiliser le code 
suivant : 


1| Hashtable section = (Hashtable)ConfigurationManager.GetSection( 
"“InformationsUtilisateur"); 


On utilise la méthode GetSection de la classe ConfigurationManager pour obtenir 
la section dont nous passons le nom en paramètre. On reçoit en retour une table de 
hachage (« HashTable ») et nous pourrons l'utiliser de cette façon pour obtenir les 
valeurs de nos clés : 


1| Console.WriteLine(sectionl['"login"]); 
2| Console.WriteLine(section["MOTDEPASSE"]); 
3| Console.WriteLine(sectionl["age"]); 


Nous pouvons boucler sur les valeurs de la section en utilisant le code suivant : 


1| foreach (DictionaryEntry d in section) 


2| € 

3 Console.WriteLine("Clé : " + d.Key + " / Valeur : " + d. 
Value); 

al} 


Le type NameValueSectionHandler 


La section de type NameValueSectionHandler ressemble beaucoup à la section précé- 
dente. Observons la configuration suivante : 


1| <configuration> 
2 <configSections?> 
3 <section name="InformationsUtilisateur" type="System. 
Configuration.NameValueSectionHandler" /> 
</configSections> 
<InformationsUtilisateur> 


<add key="motdepasse" value="12345" /> 
<add key="age" value="30" /> 
</InformationsUtilisateur?> 
10] </configuration> 


4 
5 
6 <add key="login" value="nico" /> 
7 
8 
9 


C’est la même que précédemment, à l’exception du type de la section, qui cette fois-ci 
est NameValueSectionHandler. Ce qui implique que nous obtenons un type différent 
en retour de l’appel à la méthode GetSection, à savoir un NameValueCollection : 
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1| NameValueCollection section = (NameValueCollection) 
ConfigurationManager .GetSection("InformationsUtilisateur"); 


La récupération des informations de configuration se fait de la même façon : 


1| Console.WriteLine(sectionl"login"]); 
2| Console.WriteLine (section["MOTDEPASSE"]); 
3| Console.WriteLine(sectionl['"age"]); 


Ou encore avec une boucle pour toutes les récupérer : 


1| foreach (string cle in section) 


2! À 

3 Console.WriteLine("Clé : " + cle + " / Valeur : " + 
section[clel); 

al } 


À vous de voir laquelle des deux méthodes vous préférez, mais dans tous les cas, il 
faudra fonctionner avec un système de clé / valeur. 


Le type SingleTagSectionHandler 


Ce troisième type permet de gérer une section différente des deux précédentes. Il sera 
possible d’avoir autant d’attributs que l’on souhaite dans la section. Prenez par exemple 
cette configuration : 


1| <configuration?> 

2 <configSections?> 

3 <section name="MonUtilisateur" type="System.Configuration. 
SingleTagSectionHandler" /> 

</configSections> 

5 <MonUtilisateur prenom="Nico" age="30" adresse="9 rue des 

bois"/> 
6| </configuration> 


EN 


Nous voyons que je peux mettre autant d’attributs que je le souhaite. Par contre, il 
ne sera possible de faire apparaître la section MonUtilisateur qu’une seule fois, alors 
que dans les sections précédentes, nous avions une liste de clé / valeur. Nous pourrons 
récupérer notre configuration avec le code suivant : 


1| Hashtable section = (Hashtable)ConfigurationManager.GetSection( 
"MonUtilisateur"); 
Console.WriteLine(sectionl'"prenom"l]); 
Console.WriteLine(sectionl['"age"]); 
Console.WriteLine (sectionl"'adresse"]); 


Attention par contre, cette fois-ci la casse est importante pour obtenir la valeur de notre 


attribut. Notons notre boucle habituelle permettant de retrouver tous les attributs de 
notre section : 
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1| foreach (DictionaryEntry d in section) 


2| € 

3 Console.WriteLine ("Attribut : "+ d.Key + 
d.Value); 

al} 


Voilà pour les sections utilisant un type prédéfini. 


Les groupes de sections 


/ Valeur 


Super, nous savons définir des sections de configuration. Elles nous permettent d’or- 
ganiser un peu mieux notre fichier de configuration. Par contre, si les sections se mul- 
tiplient, cela va à nouveau être le bazar. Heureusement, les groupes de sections sont 
là pour remettre de l’ordre. Comme son nom l’indique, un groupe de sections va per- 
mettre de regrouper plusieurs sections. Le but est de clarifier le fichier de configuration. 


Regardons l’exemple suivant : 


1] <configuration> 

2 <configSections?> 

3 <sectionGroup name="Utilisateur"> 
4 


<section name="ParametreConnexion" type="System. 


Configuration.SingleTagSectionHandler" 


5 <section name="InfoPersos" type="System.Configuration. 
DictionarySectionHandler" /> 

6 </sectionGroup> 

7 </configSections> 

8 

9 <Utilisateur?> 

10 <ParametreConnexion Login="Nico" MotDePasse="12345" 
Authentification Locale"/> 

11 <InfoPersos?> 

12 <add key="prenom" value="Nicolas" /> 

13 <add key="age" value="30" /> 

14 </InfoPersos> 

15 </Utilisateur> 


16| </configuration> 


/> 


Mode=" 


Nous voyons ici que j'ai défini un groupe qui s’appelle Utilisateur, en utilisant la 
balise sectionGroup, contenant deux sections de configuration. Remarquons plus bas 
le contenu des sections et nous remarquons que la balise <Utilisateur> contient nos 
sections de configuration comme précédemment. Pour obtenir nos valeurs de configu- 
ration, la seule chose qui change est la façon de charger la section. Ici, nous mettons le 


nom de la section précédée du nom du groupe. Ce qui donne : 


1| Hashtable sectioni = (Hashtable)ConfigurationManager.GetSection 


("Utilisateur/ParametreConnexion"); 


2| Hashtable section2 = (Hashtable)ConfigurationManager.GetSection 


("Utilisateur/InfoPersos"); 
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Après, la façon de récupérer les valeurs de configuration de chaque section reste la 
même. Avouez que c’est quand même plus clair non ? 


Créer une section de configuration personnalisée 


Nous allons étudier rapidement comment créer des sections de configuration personnali- 
sées. Pour cela, il faut créer une section en dérivant de la classe ConfigurationSection. 
Cette classe permet de représenter une section d’un fichier de configuration. Donc, en 
toute logique, nous pouvons l’enrichir avec nos propriétés. Il suffit pour cela de décorer 
nos propres propriétés avec l’attribut ConfigurationProperty. Ce qui donne : 


1| public class PersonneSection : ConfigurationSection 

2| € 

3 [ConfigurationProperty("age", IsRequired = true)] 
4 public int Age 

5 { 

6 get { return (int)thisl["age"]l; } 

7 set { this["age"] = value; } 

8 } 

9 

10 [ConfigurationProperty('"prenom", IsRequired = true)] 
11 public string Prenom 

12 { 

13 get Ÿ{ return (string)this["prenom"]; } 

14 set { this["prenom"] = value; } 

15 } 


16| } 


Le grand intérêt ici est de pouvoir typer les propriétés. Ainsi, nous pouvons avoir une 
section de configuration qui travaille avec un entier par exemple. Tout est fait par 
la classe mère ici et il suffit d’utiliser ses propriétés indexées en y accédant par son 
nom. Pour que notre section personnalisée soit reconnue, il faut la déclarer avec notre 
nouveau type : 


1] <configSections?> 

2 <section name="PersonneSection" type="MaPremiereApplication. 
PersonneSection, MaPremiereApplication" /> 

3[ </configSections> 


Le nom du type est constitué du nom complet du type (espace de noms + nom de la 
classe) suivi d’une virgule et du nom de l’assembly. Ici, l’espace de noms est le même 
que l’assembly, car j'ai créé mes classes à la racine du projet. Si vous avez un doute, 
vous devez vérifier l’espace de noms dans lequel est déclarée la classe. Ensuite, nous 
pourrons définir notre section, ce qui donne au final : 


1| <configuration?> 

2 <configSections?> 

3 <section name="PersonneSection" type=" 
MaPremiereApplication.PersonneSection, 
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MaPremiereApplication " /> 
4 </configSections?> 
5 <PersonneSection prenom="nico" age="30"/> 


6| </configuration> 


Pour accéder aux informations contenues dans la section, il faudra charger la section 
comme d'habitude : 


1| PersonneSection section = (PersonneSection)ConfigurationManager 
.GetSection("PersonneSection"); 

2| Console.WriteLine(section.Prenom + " a " + section.Age + " ans" 
); 


Ce qui est intéressant de remarquer ici, c’est qu’on accède directement à nos propriétés 
via notre section personnalisée. Ce qui est une grande force et permet de travailler avec 
un entier et une chaîne de caractères. Il faudra faire attention à deux choses ici. La 
première est la casse. Comme on l’a vu dans le code, le nom est écrit en minuscule. 
Il faudra être cohérent entre le nom indiqué dans l’attribut ConfigurationProperty, 
celui indiqué en paramètre de l’opérateur d’indexation et celui écrit dans le fichier de 
configuration. Tout doit être orthographié de la même façon, sinon nous aurons une 
exception : 


Exception non gérée : System.Configuration. 
ConfigurationErrorsException: Attribut ?PRENOM’? non reconnu. 


Notez que les noms d’attributs respectent la casse. [.] \ 
MaPremiereApplication\Program.cs:ligne 14 


De même, si nous ne saisissons pas une valeur entière dans l’attribut age, il va y avoir 
un problème de conversion : 


Exception non gérée : System.Configuration. 
ConfigurationErrorsException: La valeur de la propriété ?’age”? 
ne peut pas être analysée. L’erreur est : trente n’est pas 
une valeur valide pour Int32. [.] \MaPremiereApplication\ 


Program.cs:ligne 14 


Créer une section personnalisée avec une collection 


Dans le paragraphe du dessus, on constate qu’on ne peut définir qu’un élément dans 
notre section. Il pourrait être intéressant dans certains cas d’avoir une section per- 
sonnalisée qui puisse contenir plusieurs éléments, par exemple pour avoir une liste de 
personnes. 


Pour ce faire, on utilisera la classe ConfigurationPropertyCollection. La première 
chose est de créer un élément en dérivant de la classe ConfigurationElement. Cet 
élément va ressembler beaucoup à ce qu’on a fait pour créer une section personnalisée : 
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1] public class ClientElement : ConfigurationElement 

2| € 

3 private static readonly ConfigurationPropertyCollection 

_proprietes; 

4 private static readonly ConfigurationProperty age; 

5 private static readonly ConfigurationProperty prenom; 

6 

7 static ClientElement() 

8 { 

9 prenom = new ConfigurationProperty("prenom", typeof( 
string), null, ConfigurationProperty0Options.IsKey); 

10 age = new ConfigurationProperty("age", typeof(int), 
null, ConfigurationProperty0Options.IsRequired); 

11 _proprietes = new ConfigurationPropertyCollection { 
prenom, age }; 

12 } 

13 

14 public string Prenom 

15 { 

16 get Ÿ{ return (string)this["prenom"]; } 

17 set { this["prenom"] = value; } 

18 } 

19 

20 public int Age 

21 € 

22 get { return (int)thisl["age"]l; } 

23 set { this["age"] = value; } 

24 } 

25 

26 protected override ConfigurationPropertyCollection 

Properties 

27 { 

28 get { return _proprietes; } 

29 } 

30 | } 


Ici, je définis deux propriétés, Prenom et Age, qui me permettent bien sûr d’y stocker 
un prénom et un âge qui sont respectivement une chaîne de caractères et un entier. 
À noter que nous avons besoin de décrire ces propriétés dans le constructeur statique 
de la classe. Pour cela, il faut lui indiquer son nom, c’est-à-dire la chaîne qui sera 
utilisée comme attribut dans l’élément de la section de configuration. Ensuite, nous 
lui indiquons son type; pour cela on utilise le mot-clé typeof qui permet justement 
de renvoyer le type (dans le sens objet Type) d’un type. Enfin nous lui indiquons une 
option, par exemple le prénom sera la clé de mon élément (qui est une valeur unique et 
obligatoire à saisir) et l’âge, qui sera un élément obligatoire à saisir également. Ensuite, 
nous avons besoin d'utiliser cette classe à travers une collection d'éléments. Pour ce 
faire, il faut créer une classe qui dérive de ConfigurationElementCollection : 


1| public class ClientElementCollection 
ConfigurationElementCollection 
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public override ConfigurationElementCollectionType 


CollectionType 
{ 
get { return ConfigurationElementCollectionType. 
BasicMap; } 
} 
protected override string ElementName 
{ 
get ?{ return "Client"; } 
} 


protected override ConfigurationPropertyCollection 
Properties 


€ 
get { return new ConfigurationPropertyCollection(); 
* 
public ClientElement thislint index] 
{ 
get { return (ClientElement)BaseGet(index); } 
set 
À 
if (BaseGet(index) != null) 
{ 
BaseRemoveAt (index) ; 
} 
BaseAdd(index, value); 
} 
} 


public new ClientElement this[string nom] 


{ 
get { return (ClientElement)BaseGet (nom); } 


} 
public void Add(ClientElement item) 
{ 
BaseAdd(item) ; 
} 
public void Remove (ClientElement item) 
€ 
BaseRemove (item); 
pe 
public void RemoveAt(int index) 
Ÿ 
BaseRemoveAt (index); 
} 


} 
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49 

50 public void Clear() 

51 { 

52 BaseClear); 

53 } 

54 

55 protected override ConfigurationElement CreateNewElement () 

56 { 

57 return new ClientElement(); 

58 } 

59 

60 protected override object GetElementKey( 
ConfigurationElement element) 

61 { 

62 return ((ClientElement)element) .Prenom; 

63 } 

61| } 


Vous pouvez copier ce code grâce au code web suivant : 


Copier ce code 
Code web : 854375 


Ces classes ont toujours la même structure. Ce qui est important de voir est qu’on 
utilise à l’intérieur la classe ClientElement pour indiquer le type de la collection. Nous 
indiquons également le nom de la balise qui sera utilisée dans le fichier de configuration, 
c’est la chaîne « Client » que renvoie la propriété ElementName. Enfin, j'ai la possibilité 
de définir ma clé en substituant la méthode GetElementKey. Le reste des méthodes 
appellent les méthodes de la classe mère. 


Enfin, il faut créer notre section personnalisée, qui dérive comme d’habitude de 
ConfigurationSection : 


1] public class ListeClientSection : ConfigurationSection 
2| € 
3 private static readonly ConfigurationPropertyCollection 


proprietes; 


4 private static readonly ConfigurationProperty liste; 

5 

6 static ListeClientSection() 

7 { 

8 liste = new ConfigurationProperty(string.Empty, typeof( 
ClientElementCollection), null, 
ConfigurationProperty0Options.lsRequired | 
ConfigurationProperty0Options.IsDefaultCollection); 

9 proprietes = new ConfigurationPropertyCollection { 
liste }; 

10 } 

11 

12 public ClientElementCollection Listes 

13 { 

14 get { return (ClientElementCollection)baselliste]; } 
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15 } 

16 

i7 public new ClientElement this[string nom] 

18 { 

19 get { return Listes[nom]l; } 

20 } 

21 

22 protected override ConfigurationPropertyCollection 
Properties 

23 { 

24 get { return proprietes; } 

25 } 

26| } 


Notons dans cette classe comment nous utilisons l’opérateur d'indexation pour renvoyer 
un élément à partir de sa clé et renvoyer la liste des éléments. Maintenant, nous pouvons 
écrire notre configuration : 


1| <configuration> 

2 <configSections?> 

3 <section name="ListeClientSection" type=" 
MaPremiereApplication.ListeClientSection, 
MaPremiereApplication" /> 

4 </configSections> 

5 <ListeClientSection> 

6 <Client prenom="Nicolas" age="30"/> 

% <Client prenom="Jérémie" age="20"/> 

8 </ListeClientSection> 

9|[ </configuration> 


Nous avons besoin à nouveau de définir la section en indiquant son type. Puis nous 
pouvons créer notre section et positionner notre liste de clients. Pour accéder à cette 
section, nous pouvons charger notre section comme avant avec la méthode GetSection : 


1| ListeClientSection section = (ListeClientSection) 
ConfigurationManager .GetSection("ListeClientSection"); 


Puis nous pouvons itérer sur les éléments de notre section : 


1] foreach (ClientElement clientElement in section.Listes) 


2! { 

8 Console.WriteLine(clientElement.Prenom + " a " + 
clientElement.Age + " ans"); 

al } 


Ou bien, nous pouvons accéder à un élément à partir de sa clé : 


1| ClientElement elementNicolas = sectionl'Nicolas"]; 

2| Console.WriteLine(elementNicolas.Prenom + " a " + 
elementNicolas.Age + " ans" 

3 

4| ClientElement elementJeremie = section["Jérémie"]; 
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5| Console.WriteLine(elementJeremie.Prenom + " a " + 


elementJeremie.Age + " ans"); 


Ce qui donnera : 


Nicolas a 30 ans 
Jérémie a 20 ans 


Voilà pour la section personnalisée avec une liste d'éléments. Cette partie était peut- 
être un peu plus compliquée, mais retenons que la structure est souvent la même, il 
sera facile d'adapter vos sections à partir de ce code. 


Nous avons découvert plusieurs façons de stocker des paramètres utilisables par notre 
application. Il faut savoir que beaucoup de composants du framework .NET sont in- 
timement liés à ce fichier de configuration, comme une application web créée avec 
ASP.NET ou lorsqu'on utilise des services web. 


Il est important de remarquer que ce fichier est un fichier de configuration d’appli- 
cation. Il y a d’autres endroits du même genre pour stocker de la configuration pour 
les applications .NET, comme le machine.config qui est un fichier de configuration 
partagé par toutes les applications de la machine. Il y a un héritage entre les différents 
fichiers de configuration. Si l’on définit une configuration au niveau machine (dans le 
machine.config), il est possible de la redéfinir pour notre application (app.config). 
En général, le fichier machine.config se trouve dans le répertoire d'installation du 
framework .NET, c’est-à-dire dans un sous répertoire du système d’exploitation, dé- 
pendant de la version du framework .NET installée. Chez moi par exemple, il se trouve 
dans le répertoire : C:\Windows\Microsoft.NET\Framework\v4.0.30319\Config. 


Enfin, remarquons qu’il est possible de faire des modifications du fichier de configuration 
directement depuis le code de notre application. C’est un point qui est rarement utilisé 
et que j'ai choisi de ne pas présenter pour que le chapitre ne soit pas trop long! 


En résumé 


— Les fichiers de configuration sont des fichiers XML qui possèdent les paramètres de 
configuration de notre application. 

— Pour les applications console, ils s'appellent app.config. 

— On peut définir toutes sortes de valeurs de configuration grâce aux sections prédéfi- 
nies ou en ajoutant son propre type de section personnalisée. 
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Introduction à LINQ 


Difficulté : CB 


INQ signifie Language INtegrated Query. C'est un ensemble d'extensions du langage 

permettant de faire des requêtes sur des données en faisant abstraction de leur type. 

Il permet d'utiliser facilement un jeu d'instructions supplémentaires afin de filtrer des 
données, faire des sélections, etc. Il existe plusieurs domaines d'applications pour LINQ : 


— Linq To Entities ou Linq To SQL qui utilisent ces extensions de langage sur les bases de 
données. 

— Linq To XML qui utilise ces extensions de langage pour travailler avec les fichiers XML. 

— Linq To Object qui permet de travailler avec des collections d'objets en mémoire. 


L'étude de LINQ nécessiterait un livre en entier, aussi nous allons nous concentrer sur 
la partie qui va le plus nous servir en tant que débutant et qui va nous permettre de 
commencer à travailler simplement avec cette nouvelle syntaxe, à savoir Linq To Object. Il 
s'agit d'extensions permettant de faire des requêtes sur les objets en mémoire et notamment 
sur toutes les listes ou collections. En fait, sur tout ce qui implémente IEnumerable<». 
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Les requêtes Linq 


Les requêtes Linq proposent une nouvelle syntaxe permettant d’écrire des requêtes qui 
ressemblent de loin à des requêtes SQL. Pour ceux qui ne connaissent pas le SQL, il 
s’agit d’un langage permettant de faire des requêtes sur les bases de données. Pour 
utiliser ces requêtes, il faut ajouter l’espace de noms adéquat, à savoir : 


1 | using System.Ling; 


Ce using est en général inclus par défaut lorsqu'on crée un nouveau fichier. 


Jusqu'à maintenant, si nous voulions afficher les entiers d’une liste d’entiers qui sont 
strictement supérieurs à 5, nous aurions fait : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| foreach (int i in liste) 

31 À 

4 1Ë Ci > 5) 

5 £ 

6 Console.WriteLine(i); 

7 } 

8| } 


Grâce à Ling To Objet, nous allons pouvoir filtrer en amont la liste afin de ne parcourir 
que les entiers qui nous intéressent, en faisant : 


List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
IEnumerable<int> requeteFiltree = from i in liste 

where i > 5 

select ji; 


{ 


Eh 
2 
3 
4 
5| foreach (int i in requeteFiltree) 
6 
té Console.WriteLine(i); 

8 


} 


ce qui donnera : 


Nous avons ici créé une requête Linq qui contient des mots-clés, comme from, in, where 
et select. Ici, cette requête veut dire que nous partons (from) de la liste «liste » en 
analysant chaque entier de la liste dans (in) la variable i où (where) i est supérieur à 
5. Dans ce cas, l’entier est sélectionné comme faisant partie du résultat de la requête 
grâce au mot-clé select, qui fait un peu office de return. Cette requête renvoie un 
IEnumerable<int>. Le type générique est ici le type int car c’est le type de la variable 
i qui est retourné par le select. Le fait de renvoyer un IEnumerable<> va permettre 
potentiellement de réutiliser le résultat de cette requête pour un filtre successif ou une 
expression différente. En effet, Linq travaille sur des TEnumerable<>. Nous pourrions 
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par exemple ordonner cette liste par ordre croissant grâce au mot-clé orderby. Cela 
donnerait : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| IEnumerable<int> requeteFiltree = from i in liste 

3 where i > 5 

4 select ji; 

5| IEnumerable<int> requeteUrdonnee = from i in requeteFiltree 
6 orderby ïi 

7 select ji; 

8| foreach (int i in requeteUrdonnee) 

9 + 

10 Console.WriteLine(i); 

ul} 


qui pourrait également s’écrire en une seule fois avec : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| IEnumerable<int> requete = from i in liste 

3 where i > 5 

4 orderby ïi 

5 select ji; 

6| foreach (int i in requete) 

7| t 

8 Console.WriteLine(i); 

9] } 


Et nous aurons : 


H © © 


L'intérêt est que grâce à ces syntaxes, nous pouvons combiner facilement plusieurs 
filtres et construire des requêtes plus ou moins complexes. Par exemple, imaginons des 
clients : 


1| public class Client 

2| € 

3 public int Identifiant { get; set; } 
4 public string Nom { get; set; } 

5 public int Age { get; set; } 

6| } 


dont ou voudrait savoir s’ils sont majeurs, puis qu’on voudrait trier par Age puis par 
Nom, nous pourrions faire : 


1| List<Client> listeClients = new List <Client > 

2! { 

3 new Client { Identifiant = 1, Nom = "Nicolas", Age = 30}, 
4 new Client { Identifiant = 2, Nom = "Jérémie", Age = 20}, 
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5 new Client { Identifiant = 3, Nom = "Delphine", Age = 30}, 
6 new Client { Identifiant = 4, Nom = "Bob", Age = 10} 

7|}; 

8 

9| IEnumerable<string> requete = from client in listeClients 

10 where client.Age > 18 

11 orderby client.Age, client.Nom 
12 select client.Nom; 

13 

14| foreach (string prenom in requete) 

15| 

16 Console .WriteLine (prenom); 

17 | } 


Ce qui donnera : 


Jérémie 


Delphine 
Nicolas 


Notez ici que mon select renvoie le nom du client, qui est un string. Il est donc normal 
que ma requête renvoie un IEnumerable<string> car j'ai choisi qu’elle ne sélectionne 
que les noms qui sont de type string. Il est assez fréquent de construire des objets 
anonymes à l’intérieur d’une requête. Par exemple, plutôt que de renvoyer uniquement 
le nom du client, je pourrais également renvoyer l’âge, mais comme je n’ai pas besoin 
de l'identifiant, je peux créer un objet anonyme juste avec ces deux propriétés : 


1|[ List<Client> listeClients = new List <Client > 

2| { 

3 new Client { Identifiant = 1, Nom = "Nicolas", Age = 30}, 

4 new Client { Identifiant = 2, Nom = "Jérémie", Age = 20}, 

5 new Client { Identifiant = 3, Nom = "Delphine", Age = 30}, 

6 new Client { Identifiant = 4, Nom = "Bob", Age = 10}, 

t| }; 

8 

9| var requete = from client in listeClients 

10 where client.Age > 18 

11 orderby client.Age , client.Nom 

12 select new { client.Nom, client 
.Age }; 

13| foreach (var obj in requete) 

14 | À 

15 Console.WriteLine("{O0} a {1} ans", obj.Nom, obj.Age); 

16| } 


Et nous aurons : 


Jérémie a 20 ans 


Delphine a 30 ans 
Nicolas a 30 ans 
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Mon objet anonyme contient ici juste une propriété Nom et une propriété Age. À noter 
que je suis obligé à ce moment-là d’utiliser le mot-clé var pour définir la requête, car 
je n’ai pas de type à donner à cette requête. De même, dans le foreach je dois utiliser 
le mot-clé var pour définir le type anonyme. Les requêtes peuvent être de plus en plus 
compliquées, comme faisant des jointures. Par exemple, ajoutons une classe Commande : 


1| public class Commande 

2| € 

3 public int Identifiant { get; set; } 

4 public int IdentifiantClient { get; set; } 
5 public decimal Prix { get; set; } 

6| } 


Je peux créer des commandes pour des clients : 


1| List<Client> listeClients = new List <Client > 

2! À 

3 new Client { Identifiant = 1, Nom = "Nicolas", Age = 30}, 

4 new Client { Identifiant = 2, Nom = "Jérémie", Age = 20}, 

5 new Client { Identifiant = 3, Nom = "Delphine", Age = 30}, 

6 new Client { Identifiant = 4, Nom = "Bob", Age = 10}, 

7 | #3 

8 

9 | List <Commande> listeCommandes = new List <Commande > 

10! { 

11 new Commande{ Identifiant = 1, IdentifiantClient = 1, Prix 
= 150.05M}, 

12 new Commande{ Identifiant = 2, IdentifiantClient = 2, Prix 
= 30M}, 

18 new Commande{ Identifiant = 3, IdentifiantClient = 1, Prix 
= 99.99M}, 

14 new Commande{ Identifiant = 4, IdentifiantClient = 1, Prix 
= 100M}, 

15 new Commande{ Identifiant = 5, IdentifiantClient = 3, Prix 
= 80M}, 

16 new Commande{ Identifiant = 6, IdentifiantClient = 3, Prix 
= 10M}, 

27) És 


Et grâce à une jointure, récupérer avec ma requête le nom du client et le prix de sa 
commande : 


1] var liste = from commande in listeCommandes 
2 join client in listeClients on commande. 
IdentifiantClient equals client.Identifiant 


3 select new { client.Nom, commande.Prix }; 

4 

5| foreach (var element in liste) 

6| À 

7 Console.WriteLine("Le client {0} a payé {1}", element.Nom, 
element.Prix); 

8| } 
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Ce qui donne : 


client Nicolas 
client Jérémie 
client Nicolas payé 99,99 
client Nicolas payé 100 
client Delphine a payé 80 
client Delphine a payé 10 


payé 150,06 
payé 30 


a 
a 
a 
a 


On utilise le mot-clé join pour faire la jointure entre les deux listes puis on utilise le 
mot-clé on et le mot-clé equals pour indiquer sur quoi on fait la jointure. À noter que 
ceci peut se réaliser en imbriquant également les from et en filtrant sur l'égalité des 
identifiants clients : 


1| var liste = from commande in listeCommandes 

2 from client in listeClients 

3 where client.Ildentifiant == commande. 

IdentifiantClient 

4 select new { client.Nom, commande.Prix }; 

5 

6| foreach (var element in liste) 

7| { 

8 Console.WriteLine("Le client {0} a payé {1}", element .Nom, 
element .Prix); 

9! } 


Il est intéressant de pouvoir regrouper les objets qui ont la même valeur. Par exemple 
pour obtenir toutes les commandes, groupées par client, on fera : 


var liste = from commande in listeCommandes 
group commande by commande.IdentifiantClient; 


1 
2 
3 
4| foreach (var element in liste) 
5 
6 


{ 
Console.WriteLine("Le client : {0} a réalisé {1} commande(s 
)", element.Key, element.Count ()); 
7 foreach (Commande commande in element) 
8 € 
Console.WriteLine("\tPrix : {0}", commande.Prix); 
10 } 
11| } 


Ici, cela donne : 


Le client : 1 a réalisé 3 commande(s) 
Prix 150,05 
Prix 99,99 
BASS O 0) 

Le client : 2 a réalisé 1 commande(s) 
PES SS 0 

Le client : 3 a réalisé 2 commande(s) 
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PSC ES 0 
honte 2 AO) 


Il est possible de cumuler le group by avec notre jointure précédente histoire d’avoir 
également le nom du client : 


1 


NN Oo & & 


Et nous obtenons : 


var liste 


from commande in listeCommandes 

join client in listeClients on commande. 
IdentifiantClient equals client.Identifiant 

group commande by new {commande.IdentifiantClient, 
client.Nom}; 


foreach (var element in liste) 


{ 


Console.WriteLine("Le client {0} ({1}) a réalisé {2} 
commande (s)", element.Key.Nom, element .Key. 
IdentifiantClient , element.Count ()); 

foreach (Commande commande in element) 


{ 


Console.WriteLine("\tPrix : {0}", commande.Prix); 


} 
} 


Le client Nicolas (1) a réalisé 3 commandes) 


1ranlse 


PAR 


PT 


150 ,05 
9999 
100 


Le client Jérémie (2) a réalisé 1 commandes) 


Pre 


30 


Le client Delphine (3) a réalisé 2 commande(s) 
Prix 


PT 


80 
10 


À noter que le group by termine la requête, un peu comme le select. Ainsi, si l’on 
veut sélectionner quelque chose après le group by, il faudra utiliser le mot-clé into et 
la syntaxe suivante : 


1 


© © 1 oO où À 


var liste 


from commande in listeCommandes 
join client in listeClients on commande. 
IdentifiantClient equals client.Identifiant 
group commande by new {commande.IdentifiantClient, 
client.Nom} into commandesGroupees 
select 
new 
+ 
commandesGroupees.Key.IdentifiantClient, 
commandesGroupees.Key.Nom, 
NombreDeCommandes = commandesGroupees.Count 


O 
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10 FS 

11 

12] foreach (var element in liste) 

13 | À 

14 Console.WriteLine('"Le client {0} ({1}) a réalisé {2} 
commande(s)", element.Nom, element.IldentifiantClient, 


element . NombreDeCommandes); 


Avec pour résultat : 


Le client Nicolas (1) a réalisé 3 commandes) 


Le client Jérémie (2) a réalisé 1 commande(s) 
Le client Delphine (3) a réalisé 2 commande(s) 


L'intérêt d'utiliser le mot-clé into est également de pouvoir enchaîner avec une autre 
jointure ou un autre filtre permettant de continuer la requête. Il est également possible 
d'utiliser des variables à l’intérieur des requêtes grâce au mot-clé let. Cela va nous 
permettre de stocker des résultats temporaires pour les réutiliser ensuite : 


1| var liste = from commande in listeCommandes 
2 join client in listeClients on commande. 
IdentifiantClient equals client.Identifiant 
3 group commande by new {commande.ldentifiantClient, 
client.Nom} into commandesGroupees 

4 let total = commandesGroupees.Sum(c => c.Prix) 

5 where total > 50 

6 orderby total 

7 select new 

8 { 

9 commandesGroupees.Key.IdentifiantClient, 

10 commandesGroupees.Key.Nom, 

11 NombreDeCommandes = commandesGroupees.Count 

O, 

12 PrixTotal = total 

13 }; 

14 

15| foreach (var element in liste) 

16| 

17 Console.WriteLine("Le client {0} ({1}) a réalisé {2} 
commande(s) pour un total de {3}", element .Nom, element. 
IdentifiantClient , element .NombreDeCommandes, element. 
PrixTotal); 

18| } 


Par exemple, ici j'utilise le mot-clé let pour stocker le total d’une commande groupée 
dans la variable total (nous verrons la méthode Sum() un tout petit peu plus bas), 
ce qui me permet ensuite de filtrer avec un where pour obtenir les commandes dont le 
total est supérieur à 50 et de les trier par ordre de prix croissant. 


Ce qui donne : 
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Le client Delphine (3) a réalisé 2 commande(s) pour un total de 
90 


Le client Nicolas (1) a réalisé 3 commande(s) pour un total de 
350 ,04 


Nous allons nous arrêter là pour cet aperçu des requêtes LINQ. Nous avons pu voir 
que le C7 dispose d’un certain nombre de mots-clés qui permettent de manipuler nos 
données de manière très puissante mais d’une façon un peu inhabituelle. 


Cette façon d'écrire des requêtes LINQ s'appelle en anglais la sugar syntax, 
que l'on peut traduire par « sucre syntaxique ». || désigne de manière générale 
les constructions d'un langage qui facilitent la rédaction du code sans modifier 
l'expressivité du langage. 


Voyons à présent ce qu’il y a derrière cette jolie syntaxe. 


Les méthodes d’extension Linq 


En fait, toute la sugar syntax que nous avons vue précédemment repose sur un certain 
nombre de méthodes d’extension qui travaillent sur les types IEnumerable<T>. Par 
exemple, la requête suivante : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| IEnumerable<int> requeteFiltree = from i in liste 

3 where i > 5 

4 select ji; 

5| foreach (int i in requeteFiltree) 

6| { 

f Console.WriteLine(i); 

8| } 


s'écrit véritablement : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| IEnumerable<int> requeteFiltree = liste.Where(i => i > 5); 

3| foreach (int i in requeteFiltree) 

alt 

5 Console.WriteLine(i); 

6| } 


Nous utilisons la méthode d’extension Where () en lui fournissant une expression lambda 
servant de prédicat pour filtrer la liste. C’est de cette façon que le compilateur traduit 
la sugar syntax. Ce n’est donc qu’une façon plus élégante d'utiliser ces méthodes d’ex- 
tension. Chaque méthode d’extension renvoie un IEnumerable<T> ce qui permet d’en- 
chaîner facilement les filtres successifs. Par exemple, rajoutons une date et un nombre 
d'articles à notre classe Commande : 
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1| public class Commande 

2| { 

3 public int Identifiant { get; set; } 

4 public int IdentifiantClient { get; set; } 
5 public decimal Prix { get; set; } 

6 public DateTime Date { get; set; } 

7 public int NombreArticles { get; set; } 

8| } 


Avec la requête suivante : 


IEnumerable <Commande> commandesFiltrees = listeCommandes. 
Where (commande => commande.Prix > 100). 

Where (commande => commande.NombreArticles > 10). 

OrderBy (commande => commande.Prix). 

ThenBy(commande => commande .DateAchat); 


OO & © NN H 


nous pouvons obtenir les commandes dont le prix est supérieur à 100, où le nombre 
d'articles est supérieur à 10, triées par prix puis par date d’achat. 


De plus, ces méthodes d’extension font beaucoup plus de choses que ce que l’on peut 
faire avec la sugar syntax. Il existe pas mal de méthodes intéressantes, que nous ne 
pourrons pas toutes étudier. Regardons par exemple la méthode Sum() (qui a été utilisée 
dans le paragraphe précédent) qui permet de faire la somme des éléments d’une liste 
ou la méthode Average() qui permet d’en faire la moyenne : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| Console.WriteLine ("Somme : {0}", liste.Sum()); 
3| Console.WriteLine ("Moyenne : {0}", liste.Average()); 


Qui nous renvoie dans ce cas : 


Somme : bi 


Moyenne : 6,375 


Tout est déjà fait, c’est pratique ! Évidemment, les surcharges de ces deux méthodes 
d'extension ne fonctionnent qu'avec des types int ou double ou decimal... Qui envi- 
sagerait de faire une moyenne sur une chaîne ? Par contre, il est possible de définir une 
expression lambda dans la méthode Sum() afin de faire la somme sur un élément d’un 
objet, comme le prix de notre commande : 


1| decimal prixTotal = listeCommandes.Sum(commande => commande. 


Prix); 


D'autres méthodes sont bien utiles. Par exemple la méthode d'extension Take() nous 
permet de récupérer les N premiers éléments d’une liste : 


1| IEnumerable<Client> extrait = listeClients.0rderByDescending( 


client => client .Age).Take(5); 


Ici, je trie dans un premier temps ma liste par âge décroissant, et je prends les 5 
premiers. Ce qui signifie que je prends les 5 plus vieux clients de ma liste. Et s’il n’y 
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en a que 3? eh bien il prendra uniquement les 3 premiers! Suivant le même principe, 
on peut utiliser la méthode First() pour obtenir le premier élément d’une liste : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 

2| int premier = liste.Where(i => i > 5).First(); 

J'obtiens alors le premier élément de la liste qui est strictement supérieur à 5. À noter 
que le filtre peut également se faire dans l’expression lambda de la méthode First 0 : 
1 | int premier = liste.First(i => i > 5); 


Ce qui revient exactement au même. Attention, s’il n’y a aucun élément dans la liste, 
alors la méthode First() lève l'exception : 


Exception non gérée : System.InvalidOperationException: La sé 


quence ne contient aucun élément. 


Il est possible dans ce cas-là d'éviter une exception avec la méthode FirstOrDefault() 
qui renvoie la valeur par défaut du type de la liste (0 si c’est un type valeur, null si 
c’est un type référence) : 


1] Client nicolas = listeClients.FirstOrDefault(client => client. 
Nom == "Nicolas"); 

2| if (nicolas == null) 

3 Console.WriteLine("Client non trouvé"); 


Ici, je cherche le premier des clients dont le nom est Nicolas. S'il n’est pas trouvé, alors 
FirstOrDefault() me renvoie null, sinon, il me renvoie bien sûr le bon objet Client. 
Dans le même genre, nous pouvons compter grâce à la méthode Count () le nombre 
d'éléments d’une source de données suivant un critère : 


1| int nombreClientsMajeurs = listeClients.Count (client => client. 


Age >= 18); 


Ici, j'obtiendrai le nombre de clients majeurs dans ma liste. De la même façon qu’avec la 
sugar syntar, il est possible de faire une sélection précise des données que l’on souhaite 
extraire, grâce à la méthode Select () : 


1| var requete = listeClients.Where(client => client.Age >= 18). 


Select(client => new { client.Age, client.Nom }) ; 


Cela me permettra d'obtenir une requête contenant les clients majeurs. À noter que 
seront retournés des objets anonymes possédant une propriété Age et une propriété 
Nom. Bien sûr, nous retrouverons nos jointures avec la méthode d’extension Join() ou 
les groupes avec la méthode GroupBy(). Il existe beaucoup de méthodes d’extension et 
il n’est pas envisageable dans ce livre de toutes les décrire. Je vais finir en vous parlant 
des méthodes ToList () et ToArray() qui, comme leurs noms le suggèrent, permettent 
de forcer la requête à être mise dans une liste ou dans un tableau : 


1| List <Client> lesPlusVieuxClients = listeClients. 
OrderByDescending(client => client.Age).Take(5).ToList(); 
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ou 


1| Client [] lesPlusVieuxClients = listeClients.0rderByDescending( 
client => client .Age).Take(5).ToArray(); 


Plutôt que d’avoir un IEnumerable<>, nous obtiendrons cette fois-ci une List<> ou un 
tableau. Le fait d'utiliser ces méthodes d’extension a des conséquences que nous allons 
décrire. 


Exécution différée 


Les méthodes d’extension LINQ ou sa syntaxe sucrée c’est bien joli, mais quel est 
l'intérêt de s’en servir plutôt que d'utiliser des boucles foreach, des if ou d’autres 
choses ? Déjà, parce qu’il y a plein de choses déjà toutes faites : la somme, la moyenne, 
la récupération de N éléments, etc. Mais aussi pour une autre raison plus importante : 
l’exécution différée. 

Nous en avons déjà parlé, l'exécution différée est possible grâce au mot-clé yield. Les 
méthodes d’extensions Linq utilisent fortement ce principe. Cela veut dire que lorsque 
nous construisons une requête, elle n’est pas exécutée tant que l’on n'itère pas sur le 
contenu de la requête. Ceci permet de stocker la requête, d’empiler éventuellement 
des filtres ou des jointures et de ne pas calculer le résultat tant qu’on n’en a pas 
explicitement besoin. 


Ainsi, imaginons que nous souhaitions trier une liste d’entiers. Avant cela, nous aurions 
fait : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2 

3| liste.Sort(); 

al liste.Add(7); 

5 

6| foreach (int i in liste) 

7| À 

8 Console.WriteLine(i); 

9! } 


Ce qui aurait affiché en toute logique la liste triée puis, à la fin, l’entier 7 rajouté, 
c’est-à-dire : 
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Avec Linq, nous allons pouvoir écrire : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2 

3| var requete = liste.0rderBy(e => e); 

al liste.Add(7); 

5 

6| foreach (int i in requete) 

7| À 

8 Console.WriteLine(i); 

9] } 


Et si nous exécutons ce code, nous aurons : 
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Bien que nous ayons ajouté la valeur 7 après avoir trié la liste avec OrderBy, on se rend 
compte que tous les entiers sont quand même triés lorsque nous les affichons. 


En effet, la requête n’a été exécutée qu’au moment du foreach. Ceci implique donc que 
le tri va tenir compte de l’ajout du 7 à la liste. La requête est construite en mémorisant 
les conditions comme notre OrderBy, mais cela fonctionne également avec un where, 
et tout ceci n’est exécuté que lorsqu'on le demande explicitement ; c’est-à-dire avec 
un foreach dans ce cas-là. En fait, tant que le C# n’est pas obligé de parcourir les 
éléments énumérables alors il ne le fait pas. Ce qui permet d’enchaîner les éventuelles 
conditions et d'éviter les parcours inutiles. Par exemple, dans le cas ci-dessous, il est 
inutile d'exécuter le premier filtre : 


1] List<int> liste = new List<int> { 4, 6, 1, 9, 5, 15, 8, 3 }; 
2| IEnumerable<int> requete = liste.Where(i => i > 5); 

3| // plein de choses qui n'ont rien à voir avec la requete 

4| requete = requete.Where(i => i > 10); 


En effet, le deuxième filtre a tout intérêt à être combiné au premier afin d’être simplifié. 
Et encore, ici, on n'utilise même pas la requête, il y a encore moins d'intérêt à effectuer 
nos filtres si nous ne nous servons pas du résultat. Ceci peut paraître inattendu, mais 
c’est très important dans la façon dont Linq s’en sert afin d’optimiser ses requêtes. Ici, 
le parcours en mémoire pourrait paraître peu coûteux, mais dans la mesure où Linq 
doit fonctionner aussi bien avec des objets, qu'avec des bases de données ou du XML 
(ou autres...), cette optimisation prend tout son sens. 


Le maître mot est la performance, primordial quand on accède aux bases de données. 
Cette exécution différée est gardée pour le plus tard possible. C'est-à-dire que le fait 
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de parcourir notre boucle va obligatoirement entraîner l’évaluation de la requête afin 
de pouvoir retourner les résultats cohérents. Il en va de même pour certaines autres 
opérations, comme la méthode Sum(). Comment pourrions-nous faire la somme de 
tous les éléments si nous ne les parcourons pas ? C’est aussi le cas pour les méthodes 
ToList() et ToArray(). Par contre, ce n’est pas le cas pour les méthodes Where, ou 
Take, etc. 


Il est important de connaître ce mécanisme. L’exécution différée est très puissante 
et connaître son fonctionnement permet de savoir exactement ce que nous faisons et 
pourquoi nous pouvons obtenir parfois des résultats étranges. 


Récapitulatif des opérateurs de requêtes 


Pour terminer avec Linq, voici un tableau récapitulatif des différents opérateurs de 
requête. Nous ne les avons pas tous étudiés ici car cela serait bien vite lassant. Mais 
grâce à leurs noms et leurs types, il est assez facile de voir à quoi ils servent afin de les 


utiliser dans la construction de nos requêtes. 


Type Opérateur de requête Exécution 
différée 

Tri des données OrderBy, OrderByDescending, ThenBy, | Oui 
ThenByDescending, Reverse 

Opérations ensemblistes Distinct, Except, Intersect, Union Oui 

Filtrage des données OfType, Where Oui 

Opérations de quantifica- | A11, Any, Contains Non 

teur 

Opérations de projection Select, SelectMany Oui 

Partitionnement des don- | Skip, SkipWhile, Take, TakeWhile Oui 

nées 

Opérations de jointure Join, GroupJoin Oui 

Regroupement de données GroupBy, ToLookup Oui 

Opérations de génération DefaultIfEmpty, Empty, Range, Repeat | Oui 

Opérations d'égalité SequenceEqual Non 

Opérations d’élément ÉlementAt, ÉlementAtOrDefault, | Non 
First, FirstOrDefault, Last, 
LastOrDefault, Single, 
SingleOrDefault 

Conversion de types de don- | AsEnumerable,  AsQueryable, Cast, | Non 

nées OfType, ToArray, ToDictionary, 
ToList, ToLookup 

Opérations de concaténa- | Concat Oui 

tion 

Opérations d’agrégation Aggregate, Average, Count, LongCount, | Non 
Max, Min, Sum 
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N'hésitez pas à consulter la documentation de ces méthodes d’extension ou à aller 
voir des exemples sur internet. Il y a beaucoup de choses à faire avec ces méthodes. 
Il est important également de bien savoir les maîtriser afin d’éviter les problèmes de 
performance. En effet, l'évaluation systématique des expressions peut être coûteuse, 
surtout quand le tout est imbriqué dans des boucles. À utiliser judicieusement ! 


Voilà pour ce petit aperçu de Linq. Rappelez-vous bien que Linq est une abstraction qui 
permet de manipuler des sources de données différentes. Nous avons vu son utilisation 
avec les objets implémentant IEnumerable<>, avec ce qu’on appelle Linq To Objects. 
Il est possible de faire du Linq en allant manipuler des données en base de données, 
on utilisera pour cela Ling To SQL ou Ling To Entity. De même, il est possible de 
manipuler les fichiers XML avec Linq To XML. 


Linq apporte des méthodes d'extension et une syntaxe complémentaire afin d’être effi- 
cace avec la manipulation de sources de données. 


Sachez enfin qu'il est possible de requêter n’importe quelle source de données à partir 
du moment où un connecteur spécifique a été développé. Cela a été fait par exemple 
pour interroger Google ou Amazon, mais aussi pour requêter sur active directory, ou 


JSON, etc. 


En résumé 


— Linq consiste en un ensemble d'extensions du langage permettant de faire des re- 
quêtes sur des données en faisant abstraction de leur type. 

— Il existe plusieurs domaines d'applications de Linq, comme Ling to Object, Ling to 
Sal, etc. 

— La sugar syntax ajoute des mots-clés qui permettent de faire des requêtes qui res- 
semblent aux requêtes faites avec le langage SQL. 

— Derrière cette syntaxe se cache un bon nombre de méthodes d’extension qui tirent 
parti des mécanismes d’exécution différée. 
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| 


Accéder aux données avec Entity 
Framework 


Difficulté : 5:33 


ous allons voir dans ce chapitre comment connecter nos applications à une base 

de données. Bien qu'il soit possible de travailler avec quasiment n'importe quelle 

base de données (oracle, mysql,...), le framework .NET offre toute sa puissance en 
fonctionnant avec SQL Server. 


Si vous vous souvenez, lors de l'installation de Visual CH Express, nous avons également 
installé Microsoft SQL Server 2008 express Service Pack 1. SQL Server 2008 Express est 
un moteur de base de données light, idéal pour travailler en local sur son PC. Dans un 
environnement de production, nous aurons tout intérêt à travailler avec la version complète 
de SQL Server, mais pour ce livre, la version express est amplement suffisante. En plus... 
elle est gratuite! 
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Les bases de données et la modélisation 


Une base de données est un gros espace de stockage structuré qui permet d’enregistrer 
efficacement de très grandes quantités d'informations. C’est un élément incontournable 
dans tout système informatique. Dans une base de données, on peut stocker des in- 
formations unitaires, comme le nom d’un client, son âge, son adresse, etc. Ce sont les 
champs. Ces champs sont regroupés sémantiquement dans des tables. Par exemple, la 
table des clients contient les informations relatives au client Nicolas, au client Jérémie, 
etc. Ces informations sont appelées des enregistrements. 


Lorsqu'on représentera les données de notre application, nous allons avoir besoin de 
plusieurs tables. Par exemple, pour représenter les données d’une application de com- 
merce, nous pouvons avoir une table « Produits », une table « Rayons », une table 
« Clients », une table & Commandes », etc. Il peut y avoir des relations entre les tables, 
par exemple un rayon peut contenir de 0 à N produits, un client peut passer de 0 à N 
commandes, etc. 


On appelle ces bases de données des bases de données relationnelles. 


Nous pouvons lire le contenu des tables et insérer de nouvelles valeurs grâce au langage 
SQL. Nous allons très peu nous en servir mais il est la base de tout requêtage en base 
de données. Nous n’allons pas faire ici de cours sur le SQL mais nous allons l’utiliser à 
certains endroits, par souci de simplicité. 


La première chose à faire est de réfléchir au modèle de données dont on aura besoin 
dans notre application. Pour ce chapitre, nous allons prendre pour exemple une ap- 
plication de commerce, dans le genre site d’e-commerce, spécialisée dans la création 
d'applications console. Nous allons commencer par modéliser les rayons et les produits. 
On a dit qu’un rayon pouvait être composé de 0 à N produits. De même, un produit 
peut appartenir à 0 ou à N rayons. Un rayon possède un identifiant, un nom et une 
description. Un produit possède un identifiant, un nom, un prix, un stock et une url 
vers son image (voir la figure 37.1). 


Urlimage 


FIGURE 37.1 —- On appelle ce schéma un modèle conceptuel de données 


Entity Framework et le mapping objet relationnel 


Finalement, la modélisation que nous venons de faire ressemble beaucoup à la modéli- 
sation orientée objet que nous commençons à maîtriser. On pourrait très bien avoir des 
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objets Produit, des objets Rayon. Un rayon pourrait contenir des produits, etc. Ne 
pourrions-nous pas essayer de représenter la base de données sous la forme d’un modèle 
orienté objet plutôt qu’un modèle relationnel ? C’est le principe de ce que l’on appelle 
un ORM ! que l’on traduit en français par « mapping objet-relationnel ». L'ORM est 
un outil qui permet de générer une couche d’accès aux données orientée objet à partir 
d’une base de données relationnelle en définissant des correspondances entre cette base 
de données et des objets. Il existe plusieurs outils d'ORM pour .NET permettant de 
générer des objets C#, comme nhibernate, Entity Framework, etc. 


Nous allons utiliser ici Entity Framework. C’est l'ORM de Microsoft. Il est totalement 
intégré à Visual C# Express et aux autres versions. Son travail consiste, entre autres, 
à : 

— modéliser ses données et générer la base correspondante ; 

— générer un modèle à partir d’une base de données existante ; 

— gérer tous les accès à la base de données (lecture, écriture, suppression....). 


Le grand intérêt est que nous allons travailler directement avec des objets et c’est lui 
qui s’occupera de tout ce qui est persistance dans la base de données. 


Allez, fini le blabla, passons à son utilisation ! Commençons par créer une application 
console MonApplicationBaseDeDonnées, ainsi qu'indiqué à la figure 37.2. 


== 
| Rechercher Mods instal > 
LA svoiaionVinéon Fame au 
CF avricionure Vaud 
mA Applition console Visual C# 
EF svicnèque de cases Vaud 
FA appication de navigateur PF Vaud + 
T1 rroieviée Maude 


Nom: MonâpplictionBaseDeDonnees| 


FIGURE 37.2 — Création de l’application console 


À présent, ajoutons un nouvel élément de type ADO.NET Entity Data Model à notre 
projet, que l’on va appeler ModelCommerce .edmx (voir figure 37.3). 


C’est ce type de fichier qui va nous permettre de modéliser nos données. L’assistant 
s'ouvre et nous choisissons Modèle vide, comme à la figure 37.4. 


On voit apparaître une nouvelle fenêtre intitulée Entity Data Model Designer. Cette 
fenêtre « designer » va nous permettre de modéliser nos données. Elle nous donne 
également accès à la boîte à outils, ainsi que vous pouvez le voir à la figure 37.5. 


Lorsque nous accédons à la boîte à outils, nous pouvons voir plusieurs éléments. Celui 
qui nous intéresse est l'entité (voir figure 37.6). 


1. C’est l’acronyme de l’expression anglaise : object relationnal mapping. 
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juMeies es Trierpar: [Par défaut] Rechercher Modèles installés P 
Éléments Visual C & FE 
+ Éléments Visual CE 
EE] cs élément Visual Ca [192 Éléments Visual C 
= Élément de projet pour réer un ADO.NET 
, || Ent Det Mode 
Je mere Éléments Visual 
=] Windows Form Éléments Visual C# |! 
ER Éléments Visual + 
@=]| Contrôle utilisateur (WPF) Éléments Visual C# | Al 
& ADONET Enity Data Model Éléments Visual C# 
& ADO.NET EntityOhject Generator Éléments Visual C# 
LR 55e de données basée suressenices Éléments Visual C# 
LR sasecedoméssiote Éléments Visual C# 
Boîte de dialogue À propos de Éléments Visual C# 
FU, Classes LINQ to SQL éléments Visual C# + 
Nom: ModelCommerce.edme | 


FIGURE 37.3 — Ajout d’un fichier ADO.NET Entity Data Model 


Générer à partir 
de la base de … 


Crée un modèle vide comme point de départ pour concevoir visuellement un modèle conceptuel à partir de 
la boîte à outils. Les classes sont générées à partir du modèle lors de la compilation du projet. Vous pouvez 
spécifier une connexion de base de données ultérieurement pour associer le modèle conceptuel au modèle 
de stockage. 


FIGURE 37.4 - Choix d’un modèle vide 


Fichier Edition Affichage Projet Déboguer Données Outis Fenêtre 2 : : 
EELTOPEEEMREDOUE EEE 


syano € 27/09 À 


Entity Data Model Designer vous permet de visualiser et concevoir votre Entity Data Model. 


Créez de nouvelles entités dans le modèle en faisant glisser les éléments à partir de Boîte à outils, 


saauu0p ap 5221n0S 1 


FIGURE 37.5 — La boîte à outil est accessible sur la gauche 
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Fichier Edition Affichage Projet Déboguer Données Outi 
iddJ-sdd|k3m|9-e-|r ||. 


| 4 Entity Framework | 
R Pointeur 
Association 


% Entité 
D Héritage 
4 Général 
Il n'existe aucun contrôle utilisable dans ce groupe. 


Faites glisser un élément ici pour l'ajouter à la boîte 
à outils. 


saguuop ap s221n0$ gi |sjno e 2109 x 


FIGURE 37.6 — Ajout d’une entité sur le designer 


Nous allons pouvoir modéliser nos entités en les faisant glisser sur le designer. Comme 
l'illustre la figure 37.7, nous la voyons apparaître sur le designer. 


2: Entitél Eh 


[5 propriétés 
E in 


IË © Propriétés de navig.. | 


FIGURE 37.7 — L’entité apparaît sur le designer 


Nous pouvons renommer cette entité, soit en allant modifier la propriété Nom dans 
la fenêtre de propriétés, soit en cliquant directement sur le nom dans le designer. 
Appelons cette entité Rayon. Nous pouvons constater que cette entité est générée avec 
une propriété par défaut : ID. C’est l'identifiant de notre rayon. Cet identifiant possède 
des propriétés que nous pouvons voir dans la fenêtre de propriétés (voir figure 37.8). 


Nous pouvons par exemple voir (ou modifier) son nom (ID) et son type (Int32, qui 
est l'équivalent de int). Nous voyons également que cette propriété est la clé d’entité. 
Ce qui veut dire que c’est ce qui va nous permettre d'identifier notre rayon de manière 
unique (c’est un peu plus complexe que ça, mais retenons ce point). Nous pouvons 
ajouter une nouvelle propriété à l'entité; pour cela il suffit de faire un clic droit sur 
l'entité et d'ajouter une propriété scalaire (voir figure 37.9). 


Une nouvelle propriété apparaît dans notre entité Rayon (voir figure 37.10). 


Nous pouvons la renommer en Nom. Nous pouvons voir sur la figure 37.11 qu’elle est 
par défaut du type String, ce qui nous va très bien. 
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ModelCommerce.Rayon.ID Property 


D 


Clé d'entité 
Documentation 

Getter Public 
Mode d'accès conct None 


Mappage de table 
|E4 Mappage de procédure stockée 


Valider 
13 Propriétés 


Mettre à jour le modèle à partir de la base de données... 
Générer la base de données à partir du modèle... 
Ajouter un élément de génération de code... 


Propriété de navigation 


Propriété complexe 


FIGURE 37.9 — Ajout d’une propriété scalaire 


& Property 


E Propriétés de navig.… 


Propriétés 


2211 


Longueur fixe (Aucun) 
Longueur max. (Aucun) 


Setter Public 
StoreGeneratedPatte None 
Type String 
Unicode (Aucun) 


Nom de la propriété. 


FIGURE 37.10 —- Changement du nom de la propriété 
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1 


Longueur max. (Aucun) 


Nullable 


Unicode 
Valeur par défaut 


Type 
Type de la propriété scalaire. 


FIGURE 37.11 - Le nom et le type de la propriété 


Rajoutons une troisième propriété Description, toujours de type String mais qui 
pourra être nulle, il suffit de déclarer la propriété Nullable à True (voir la figure 37.12). 


ss. ete 


ModelCommerce.Rayon.Description Property + 


E Propriétés GE A 
St) ne 2] = 


À Nom Getter Public 
PP Description 
El Propriétés de navig.… 


Longueur fixe (Aucun) 
Longueur max. (Aucun) 
Mode d'accès conct None 


Nom Description 


Nullable True 


er Public 


StoreGeneratedPatte None 
Type String 
Unicode (Aucun) 
Valeur par défaut (Aucun) 


Nullable 
Détermine si la propriété est Nullable. 


FIGURE 37.12 — La propriété peut être nulle 


Ça y est, notre entité Rayon est modélisée! Rajoutons maintenant une nouvelle entité, 
Produit, qui possède également un identifiant, un nom de type String et un prix de 
type Decimal. Pour mettre le type à Decimal, il suffit de changer le type dans la fenêtre 
des propriétés (voir la figure 37.13). 


Ensuite, rajoutons une propriété Stock de type Int32 et une propriété UrlImage de 
type String. Nous obtenons deux superbes entités (voir la figure 37.14). 


Il est temps de relier les entités entre elles grâce à une association : faites un clic droit 
sur l’entité Rayon et ajoutez une association, comme indiqué à la figure 37.15. 
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ModelCommerce.Prod|DateTime [A] 


SE | 1 DateTimeOffset 
Getter 
Longueur fixe Guid 


Decimal 


Longueur max. Int16 
Mode d'accès conci{Int32 
Nom Int64 
Nullable SByte 
Setter Single 
StoreGeneratedPatt4 String 
String 
Unicode (Aucun) 
Valeur par défaut (Aucun) 


Type 
Type de la propriété scalaire. 


FIGURE 37.13 —- Changement du type de la propriété 


FIGURE 37.14 -— Les deux entités de notre modèle 


Propriété scalaire 
Propriété de navigation 
Propriété complexe 


Renommer 

# Couper Ctri+X 
Éà Copier Ctri+C 
# Coller Ctri+V 
X Supprimer Suppr 
Réduire 

Œ  Mappage de table 
E. 

El 


Importation de fonction... 


Mappage de procédure stockée 

Afficher dans l'explorateur de modèles 

Mettre à jour le modèle à partir de la base de données... 

Générer la base de données à partir du modèle... 

Ajouter un élément de génération de code... 

Valider 
fà Propriétés 


FIGURE 37.15 — Ajout d’une association entre les deux entités 
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Un nouvel écran s’ouvre qui permet de définir l’association. Indiquons que la multipli- 
cité est à « plusieurs » sur les deux entités, ce qui permet de dire qu’un rayon peut 
contenir de 0 à N produits et inversement, un produit peut être contenu dans 0 à N 
rayons. Notons au passage que les choix possibles sont 1, 0 ou 1 et plusieurs. C’est ce 
qui nous permet d'indiquer la cardinalité de nos relations. Changez ensuite le nom des 
propriétés de navigation en rajoutant un «s » à Produit et à Rayon, comme indiqué à 
la figure 37.16. 


Nom de l'association : 
RayonProduit 
Terminaison Terminaison 
Entité : Entité : 
| Rayon % | | Produit 
Multiphote : Muitiplicité : 
* (plusieurs) L | * (plusieurs) 
[V] Propriété de navigation : [V] Propriété de navigation : 


Rayon peut avoir * (plusieurs) instances de Produit. Utilisez Rayon.Produits 
pour accéder aux instances Produit. 


Produit peut avoir * (plusieurs) instances de Rayon. Utilisez Produit.Rayons 
pour accéder aux instances Rayon. 


FIGURE 37.16 — Changement des caractéristiques de l’association 


Le designer est mis à jour avec la relation et on peut voir apparaître des propriétés 
de navigation dans les entités. L’entité Rayon a une propriété Produits, ce qui va 
permettre d’obtenir la liste des produits d’un Rayon. De même, l'entité Produit possède 
une propriété de navigation Rayons qui va permettre d’obtenir la liste des rayons qui 
contiennent le produit (voir la figure 37.17). 


Nous allons encore faire une petite modification à ce modèle. Sélectionnez l'entité 
Rayon. Nous pouvons voir dans ses propriétés que le nom du jeu d’entité vaut RayonJeu, 
modifiez-le en Rayons, comme indiqué à la figure 37.18. 


Faites pareil pour l'entité Produit : changez ProduitJeu en Produits. 


Voilà, tout ça, c’est notre modèle! Il faut maintenant faire en sorte que notre base de 
données soit cohérente avec le modèle. Il suffit de faire un clic droit sur le designer et 
de choisir de générer la base de données à partir du modèle (voir la figure 37.19). 
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El Propriétés 
8410 
L$ Nom 


EF Description 
E Propriétés de navig… 
Æ Produits 


FIGURE 37.17 - La liaison entre les deux entités 


ModelCommerce.Rayon EntityType 


AE 
Abstrait False 
Accès Public 
b Documentation 
\om 


Nom du jeu d'entités Rayons 


Type de base (Aucun) 


Nom du jeu d'entités 
Nom du EntitySet contenant les instances de l'entité. 


FIGURE 37.18 —- Modification du nom du jeu de données 


#40 #n 
LF Nom 1 Nom 


Ajouter 

Diagramme 

Zoom 

Grille 

Format de propriété scalaire 
Sélectionner tout 


Détails de mappage 
Explorateur de modèles 


E3 à 


Valider 
23 Propriétés 


FIGURE 37.19 —- Génération de la base de données à partir du modèle 
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Une nouvelle fenêtre s'ouvre nous permettant de choisir notre source de données. Celle- 
ci étant vide, cliquez sur Nouvelle connexion (voir la figure 37.20). 


U Choisir votre connexion de données 
| 


Quelle connexion de données votre application doit-elle utiliser pour établir une connexion à la base 


0 e stockage des données 

chaîne de connexion peut entraîner un risque de sécurité. Voulez-vous inclure les données 
chaîne de connexion ? 

Non, exclure les données sensibles de la chaîne de connexion. Je définirai ces informations dans le 

code de mon application 


Oui, inclure les données sensibles dans la chaîne de connexion. 


Chaîne de connexion de l'entité : 


Enregistrer les paramètres de connexion de l'entité dans App.Config en tant que: 


< Précédent | Suivant > | Terminer 


FIGURE 37.20 — Ajouter une nouvelle connexion 


Une nouvelle fenêtre apparaît nous permettant de choisir notre source de données (voir 
la figure 37.21). 


Description 


Utilisez cette sélection pour attacher un fichier 
de base de données à une instance Microsoft 
SQL Server locale (notamment Microsoft SQL 
Express) à l'aide du fournisseur de données 
.NET Framework pour SQL Server. 


Fournisseur de données : 
[Fournisseur de données .NET Framework pour + 


Toujours utiliser cette sélection 


FIGURE 37.21 -— Choix d’un fichier de base de données 
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Attention, ici dans la version express de Visual C#, il n'est possible de choisir 
que parmi deux options : un fichier de base de données ou Microsoft SQL 
Server compact. Dans d'autres versions express (notamment la version per- 
mettant de faire du développement web) et dans les versions payantes de 
Visual Studio, il est possible de choisir directement un serveur de base de 
données. 


Cela aurait été plus pratique. Tant pis, nous allons faire avec; choïsissons le fichier 
de base de données. Il faut lui donner un emplacement et un nom, par exemple dans 
le répertoire des projets, je l'appelle basecommerce.mdf. Ensuite, pour pouvoir s’y 
connecter, nous utiliserons l’authentification Windows (voir la figure 37.22). 


pee PE _ ù 


Entrez les informations pour vous connecter à la source de 
données sélectionnée ou cliquez sur "Modifier" pour sélectionner 
une autre source de données et/ou un autre fournisseur. 


Source de données : 
Fichier de base de données Microsoft SQL Serve 


Nom de fichier de base de données (nouveau ou existant) : 


Visual Studio 2010\Projects\basecommerce/mdf 


Connexion au serveur 


@) Utiliser l'authentification Windows 
") Utiliser l'authentification SQL Server 


Nom d'utilisateu 


Mot de passe 


Enregistrer mot de passe 


FIGURE 37.22 — Sélection de la base de données et du type de connexion 


Au moment de la validation, il nous est demandé si l’on souhaite créer le fichier de base 
de données. Répondez « oui » ! 


Puis nous arrivons sur un récapitulatif et nous voyons en bas la chaîne de connexion à 
la base de données, comme l’illustre la figure 37.23. 


Nous pouvons choisir d’enregistrer les paramètres de connexion, ou bien de ne pas le 
faire, en cochant ou décochant cette case. Dans tous les cas, cette chaîne de connexion 
ne nous servira pas en l’état. Cliquez sur suivant. Le designer d’Entity Framework 
nous à finalement créé un fichier contenant des instructions SQL qu'il nous propose 
d'enregistrer. Ces instructions SQL vont permettre de générer les tables de la base 
de données (voir la figure 37.24). Nous allons revenir sur ces instructions. Ce fichier 
s'ouvre également dans Visual C;# Express (voir la figure 37.25). 
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U Choisir votre connexion de données 


Quelle connexion de données votre application doit-elle utiliser pour établir une connexion à la base 


de données ? 


ion semble contenir des données sensibles (par exemple, un mot de p. 

s sont indispensables pour établir une connexion à la base de données, Le stockage des données 
sensibles dans la chaîne de connexion peut entraîner un risque de sécurité. V 
sensibles dans la chaîne de connexion ? 


s inclure les données 


Non, exclure les données ibles de la chaîne de connexion. Je définirai ces informations dans le 
code de mon application: 


Our, inclure les donnée: 1sibles dans la chaîne de connexion 
Chaîne de connexion de l'entité : 


metadata=res://*/ModelCommerce.csdi|res://*/ModelCommerce.ssdl| 
res://*/ModelCommerce.msl:provider=System.Data.SqlClient:provider connection string="Data 
Source=.\SQLEXPRESS;:AttachDbFilename="C:\Users\Nico-PC\Documents\Visual Studio 2010\Projects 
\basecommerce.mdf";Integrated Security=True:Connect Timeout=30;User Instance=True' 


Enregistrer les paramètres de connexion de l'entité dans App.Config en tant que: 


ModelCommerceContainer 


me NS —— un 


FIGURE 37.23 — La chaîne de connexion à la base de données générée par Visual C# 
Express 


U Résumé et paramètres 


Enregistrer le DDL en tant que: ModelCommerce.edmx.sql 


Entity Designer DDL Script for SQL Server 2005, 2008, and Azure 


-- Date Created: 11/21/2011 19:25:00 
-- Generated from EDMX file: C:\Users\Nico-PC\documents\visual studio 2010\Projects\C# 


BaseDeDonnees\CoucheAccesDonnees\ModelCommerce.edmx 


SET QUOTED_IDENTIFIER OFF; 
[de] 


USE [basecommerce]; 
Go 


IF SCHEMA_ID(N'dbo') IS NULL EXECUTE(N'CREATE SCHEMA [dbol]'); 


Dropping existing FOREIGN KEY constraints 


[_ Suivant > 


FIGURE 37.24 - Visual C;} Express nous génère un script SQL 
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Fichier Edition Affichage Projet Déboguer Données Outils Fenêtre ? 
idu-E A 


1Q & E «r 


à 


-- Date Created: 11/21/2011 19:25:00 
-- Generated from EDMX file: C:\Users\Nico-PC\documents\visual s 


SET QUOTED_IDENTIFIER OFF; 

Loe] 

USE [basecommerce]; 

Go 

IF SCHEMA_ID(N'dbo') IS NULL EXECUTE(N'CREATE SCHEMA [dbo]'}; 


seauuop ap s221n0$ pk sjno e 20g à 


FIGURE 37.25 — Le script SQL s'ouvre dans Visual C# Express 


Nous en avons terminé pour l'instant avec le designer. 


Installer et utiliser l’outil de gestion de BDD 


Nous avons le serveur de base de données, qui a été installé en même temps que Visual 
C+# Express. Nous avons le script permettant de générer le modèle de données. Il nous 
manque un outil permettant de créer la base de données et d’exécuter le script. C’est 
l'outil de gestion de base de données. Je vous laisse utiliser le code web suivant pour 
installer « Microsoft®) SQL Server®) 2008 Management Studio Express » : 


Microsoft SQL Server ) 


Code web : 225473 


Notez qu'il ne s’agit que des outils puisque nous avons déjà installé un serveur de 
base de données. Il existe cependant d’autres installations qui cumulent le serveur 
de base de données ainsi que les outils. Téléchargez l’installeur qui vous convient et 
exécutez-le. Puis choisissez Installation et poursuivez. Choisissez ensuite l’ajout de 
fonctionnalités à une installation existante, comme indiqué à la figure 37.26. 


On poursuit l'installation, puis nous arrivons sur le choix des éléments à installer (voir 
la figure 37.27). 


Même si le nom est trompeur, nous devons effectuer une nouvelle installation de SQL 
Server 2008. Ce n’est pas tout à fait une installation d’une nouvelle instance car nous 
avons déjà une instance installée. Nous pouvons le voir sur la copie d'écran, où l’instance 
existante déjà installée s'appelle SQLEXPRESS. Retenons bien le nom de cette instance, 
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#5 Centre d'installation SQL Server L=-15 lis 


Nouvelle installation autonome SQL Server ou ajout de fonctionnalités à une installation 
existante 


Planification 

Lancez un Assistant permettant d'installer SQL Server 2008 dans un environnement non 
Maintenance cluster ou d'ajouter des fonctionnalités à une instance de SQL Server 2008 existante, 
Outils 
Ressources 
Avancé 
Options 


kr. Installation d'un nouveau cluster de basculement SQL Server 


Lancez un Assistant permettant d'installer un cluster de basculement SQL Server 2008 à 
un nœud. 


T Ajouter un nœud à un cluster de basculement SQL Server 


Démarrez un Assistant permettant d'ajouter un nœud à un cluster de basculement SQL 
Server 2008 existant. 


NÉ Mise à niveau de SQL Server 2000 ou SQL Server 2005 


Lancez un Assistant permettant de mettre à niveau SQL Server 2000 ou SQL Server 2005 
vers SQL Server 2008. Avant de procéder à la mise à niveau, vous devez exécuter le 
Conseiller de mise à niveau afin de détecter les problèmes potentiels. 


EN Rechercher les mises à jour du produit 
Recherchez les mises à jour du produit SQL Server 2008 dans Microsoft Update. 


PEL server200s 


FIGURE 37.26 — Choix d’une nouvelle installation 


Type d'installation 


Effectuez une nouvelle installation ou ajoutez des fonctionnalités à une instance existante de SQL Server 2008. 


Règles de support du programme … © Effectuer une nouvelle installation de SQL Server 2008 


LE ‘© Ajouter des fonctionnalités à une instance existante de SQL Server 2008 
Clé de produit (Product Key) 


Termes du contrat de licence 


[SQLEXPRESS = 


Sélection de composant 
Configuration de l'instance 
Espace disque requis 
Création de rapports d'erreurs et d'… DLEngine SQLEngine\Replication 10125310 |Express 


bnctionnalités Version Édition 


Règles d'installation 
Prêt pour l'installation 
Progression de l'installation 


Terminé 


FIGURE 37.27 - Choix du type d'installation 
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il nous servira un peu plus loin. Puis nous arrivons sur l’écran suivant qui nous indique 
que nous allons ajouter l'outil de gestion de base (voir la figure 37.28). 


[4 installation de SQL Server 2008 ES) 


Sélection de composant 


Sélectionnez les fonctionnalités de Express with Advanced Services à installer. Pour les installations en cluster, seuls les 
services Moteur de base de données et Analysis Services peuvent être en cluster. 


Règles de support du programme … | Fonctionnalités : Description : 
Aype dessein Fonctionnalités de l'instance Inclut la prise en charge de SQL Server 
Clé de produit (Product Key) Fonctionnalités partagées Management Studio pour le moteur 
Termes du contrat de licence Elo estion - De base ER ere i 
Kit de dével nt logiciel (SDK) de l'option Connect 
Sélection de composant Le se ogiciel (SDK) de l'option Comet nie SOL Server (SQLCMD) et le 
RSR nt fournisseur PowerShell SQL Server. 


Espace disque requis 

Création de rapports d'erreurs et d'… 
Règles d'installation 

Prêt pour l'installation 

Progression de l'installation 
Terminé 


«[ mt 


Sélectionner tout || Désélectionner tout | 


Répertoire des fonctionnalités partagées: |c'\Program Files\Microsoft SQL Seven, 


{ <Brécédent || Suivent> || Annuler || aide | 


FIGURE 37.28 — Fenêtre de sélection de composants à installer 


Et voilà, nous avons terminé cette installation! Maintenant, nous pouvons enfin dé- 
marrer SQL Server Management Studio! 


Au démarrage, il nous demande de nous connecter à notre instance de base de données. 
Par défaut, l’instance s'appelle SQLEXPRESS, comme nous l'avons vu, et nous pouvons 
nous y connecter en la préfixant par le nom de notre machine ou bien simplement 
en utilisant le point &.», ce qui donne .\SQLEXPRESS. Conservez l’authentification 
Windows (voir la figure 37.29). 


Vous arrivez dans l’outil et vous pouvez voir dans l’explorateur d’objets à gauche qu’il 
n’y à pas (encore) de base de données (voir la figure 37.30). 


Nous allons devoir créer notre base de données. Faites un clic droit sur le dossier Base 
de données et choisissez Nouvelle base de données. Donnez-lui le même nom que 
le fichier de base de données que nous avions précédemment créé : basecommerce. 
Validez : la base de données est créée! Nous la voyons apparaître dans l’explorateur 
d'objets (voir la figure 37.31) et nous voyons également qu’il n’y a pas de tables dedans. 


Cliquons maintenant sur Nouvelle requête, une fenêtre vide s’ouvre où nous allons 
coller le contenu du fichier SQL qui a été généré par le designer de Visual C# Express. 
Parlons un peu du contenu de ce script. Remarquons déjà que les commentaires sont 
préfixés par --, mais, malins comme nous sommes, nous les aurions reconnus, en plus 
ils sont en vert! Ensuite, la ligne suivante permet d’indiquer que nous allons nous 
positionner sur la base basecommerce : 


1 | USE [basecommercel]; 


446 


INSTALLER ET UTILISER L'OUTIL DE GESTION DE BDD 


PE SOL Server 2008 


RÉRREES [Moteur de base de données 7 
Nom du serveur : 
Authentification : [Authentification Windows 4) 
Nom d\éfisateur WIN-UOGOLE2DHAN\Nico-PC -) 
Mot de passe 
Mémoriser le mot de passe 


FIGURE 37.29 — Fenêtre de connexion à la base de données 


Fichier Edition Affichage Outils Fenêtre Co 
: ©]. Nouvelle requête BAIE AC ICCT 


Connexion + x u Ÿ TE 


SOLEXPRESS (S erver 


E T pe de die 
æ Ca Bases de données système 
Œ EL Sécurité 


D TVEUI 
a du si 
& Ca Gestion 


FIGURE 37.30 — Il n’y à aucune base de données 
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Fichier Edition Affichage Outils Fenêtre ( 
© Nouvelle requête [à [à  Æ Æ 
Connexion » | #3 4) A 
E G \SQLEXPRESS (SQL Server 10.0.2531 - WIN-U( 
[SR] Bases de données 


[4 Bases de données système 
© La émas de base de données 
E La Tables 


Œ Synonymes 
Œ Programmabilité 
a Service Broker 
Œ Sécurité 

Cm Sécurité 

Cm Objets serveur 


FIGURE 37.31 — Il n’y à pas de tables dans la base de données que nous avons créée 


Si jamais vous n’avez pas donné le même nom à la base de données, c’est ici qu’il faut 
le changer. Allons un peu plus bas, nous voyons l'instruction : 


1| CREATE TABLE [dbol.[Rayons] ( 

2 [ID] int IDENTITY(1,1) NOT NULL, 
3 [Nom] nvarchar(max) NOT NULL, 

4 [Description] nvarchar(max) NULL 
5| ); 


qui permet de créer la table contenant les rayons, suivi du même genre d'instruction 
qui permet de créer la table contenant les produits. Sans trop nous attarder dessus, 
nous pouvons voir la syntaxe permettant de créer la table (avec CREATE TABLE) et la 
syntaxe permettant de créer les champs de la table, ainsi que leurs types. 


Après la création des tables, et comme l’indiquent les commentaires pour les anglo- 
phones, la suite est une histoire de clé primaire et de clé étrangère. 


Clé primaire ? Clé étrangère ? C'est quoi ça ? 


Ce sont des notions de base de données. Sans trop entrer dans les détails, je vais vous 
expliquer rapidement de quoi il s’agit. Une clé primaire est une contrainte d’unicité 
qui permet d'identifier de manière unique un enregistrement dans une table. La clé 
primaire correspond dans notre cas à l'identifiant d’un rayon ou à l'identifiant d’un 
produit dans leurs tables respectives. À noter qu’elles ont une propriété complémen- 
taire, à savoir un auto-incrément. C'est-à-dire que c’est SQL Server qui va s'occuper 
de numéroter automatiquement ces identifiants, en les incrémentant à chaque inser- 
tion. Une clé étrangère est une contrainte qui garantit l'intégrité référentielle entre 
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deux tables. Elle identifie une colonne d’une autre table. Cela permet de faire des liens 
sémantiques entre les tables. 


Vous n'avez pas besoin de savoir exactement ce qu'il se passe dans ce script 
SQL. Nous le regardons vite fait pour la culture, mais il faut juste savoir 
l'exécuter afin qu'il nous crée les tables. 


Ce petit aparté terminé, retournons dans SQL Server Management Studio et collons-y 
notre requête. Il ne reste plus qu’à exécuter le script en cliquant sur le bouton Exécuter, 
comme indiqué à la figure 37.32. 


“el; 


QLQuery1.sql - (local)\...\Nico-P...2 


Fichier Edition Affichage Requête Débogage Outils Fenêtre Communauté 
À Nouvelle requête | [| Là | & flemme - 
ik mare 

Explorateur d'objets 
Connexion + | #4} Æ5 S 
E ($ :\SQLEXPRESS (SQL Server 10.0.2531 - WIN-UC 

€ Ca Bases de données | À __ ____________________ 
Ca Bases de données système 
E E] basecommerce 


—- Creating foreign ke; 


FIGURE 37.32 — Exécuter la requête 


Comme tout s’est bien passé, nous pouvons rafraîchir l’explorateur d’objets et constater 
que les nouvelles tables sont créées (voir la figure 37.33). 


Maintenant, nous avons besoin de données dans ces tables. Il y a plusieurs façons 
de faire. La première est d’utiliser le designer de SQL Server Management Studio, la 
seconde serait d'utiliser un script SQL, la troisième serait d'utiliser du code C#. Regar- 
dons la première solution et faisons un clic droit sur la table produit pour « modifier 
les 200 lignes du haut ». Nous pouvons ensuite insérer des valeurs, comme l’illustre la 
figure 37.34. 


Ne le faites pas, car pour vous éviter du travail, je l’ai fait pour vous grâce à la deuxième 
méthode, le script SQL : 


1| INSERT INTO [Produits] (CNoml,[Prix]l,[Stock],[Urlimage]) 
2 VALUES ('Télé HD', 299, 50, ‘tele. ps") 

3| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[UrlImagel]) 
4 VALUES ('Console de jeux', 150, 25, 'console.jpg') 
5| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[UrlImagel]) 
6 VALUES ('Canapé', 400, 10, 'canape.jpg') 

7| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[UrlImagel]) 
8 VALUES ('Cuisinière', 280, 20, ‘'cuisiniere.jpg') 

9| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[UrlImagel]) 
10 VALUES ('Bouilloire', 19, 100, l'bouilloire.jpg') 

11| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[Urllmage]) 
12 VALUES ('Lit 2 places', 149, 15, ‘lit. jpg") 

13| INSERT INTO [Produits] (CNoml,[Prix]l,[Stock],[Urlimage]) 
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Fichier Edition Affichage Débogage Outils Fenêtre Communauté Aide 

: 9 Nouvel requête | Dù | D | M à | & 

; M3 tax vl Ÿ Exécuter b 1 4 ÿ 5 E ls | 5 
SQLQuery1.sql - (local)\...\Nico-PC (.. 


E ($ :\SQLEXPRESS (SQL Server 10.0.2531 - WIN-U 
E CA Bases de données 
Bases de données système 


A -- Creating foreign key 
ées ALTER TABLE [dbo].[Rayor 
C S ADD CONSTRAINT [FK_Rayor 
Cm Tables système FOREIGN KEY ([Rayon. 
© dbo.Produits REFERENCES [dbo]. [Re 
© dbo.RayonProduit ([1D]) 
© dbo.Rayons ON DELETE NO ACTION 
Ca Vues ço 
La Synonymes 
Programmabilité —- Creating foreign key 
Service Broker ALTER TABLE [dbo].{Rayor 
Sécurité — 


Sécurité 
Objets serveur 
Réplication 

mm D Gestion 


FIGURE 37.33 — Les tables ont correctement été créées 


WIN-UOQOLE20HAN.. dbo-Produits| SQLQuepsg (ec. \Mie> PC) 

ID Nom Prix Stock Urllmage 
1 Télé HD 299 50 tele.jpg 
2 Console de jeux 150 25 console. jpg 
3 Canapé 400 10 canape.jpg 
4 Cuisinière 280 20 cuisiniere. jpg 
5 Bouilloire 19 100 bouilloire.jpg 
6 Lit 2 places 149 15 lit.jpg 

vx [AU AULL AULL MUL 


FIGURE 37.34 — Ajout de données par l’interface de SQL Server Management Studio 
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14 VALUES ('Pull', 39.99, 25, ‘pull jpg") 

15| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[Urllimagel]) 
16 VALUES ('T-shirt', 19.99, 20, “tshirt :jpg"') 

17 | INSERT INTO [Produits] ([Nom],[Prix],[Stock],[Urllmage]) 
18 VALUES ('Pyjama', 15.15, 4, 'pyjama.jpg') 

19| INSERT INTO [Produits] ([Nom],[Prix],[Stock],[Urllmage]) 
20 VALUES ('Tablette PC', 350, 44, l'tablette.jpg') 

21] INSERT INTO [Produits] ([Noml,[Prixl,[Stock],[Urllimage]) 
22 VALUES ('Smartphone', 319.99, 40, ‘smartphone. jpg') 


Il vous suffit d'exécuter ce script pour insérer les données. Nous n’allons pas détailler 
la syntaxe de ce script, mais il est quand même assez facile à lire comme ça. À noter 
que nous n’avons pas besoin d'indiquer d'identifiant car il est auto-incrémenté par SQL 
Server. Créons maintenant des rayons, avec la même technique : 


1| INSERT INTO [Rayons] ([Noml,[Descriptionl]l) 

2 VALUES ('Salon', 'Tout ce qu''on trouve dans un salon') 
3|[ INSERT INTO [Rayons] ([Noml,[Description]l) 

4 VALUES ('Cuisine', "Venez découvrir l''univers de la 


cuisine!) 
5|[ INSERT INTO [Rayons] ([Noml,[Description]) 
6 VALUES ('Dormir', null) 
7| INSERT INTO [Rayons] ([Nom]l,[Description]) 
8 VALUES ('Hi-Tech', 'Les produits hi-tech ...') 
9! INSERT INTO [Rayons] ([Noml,[Descriptionl]l) 
10 VALUES ('Vêtements', null) 


Vous pouvez copier ces codes grâce au code web suivant : 


Copier ces codes ) 


Code web : 174353 


Nous pouvons voir le contenu de ces tables en faisant un clic droit, puis Sélectionner 
les 1000 lignes du haut, comme illustré à la figure 37.35. 


Ca Bases de données 
& Ca Bases de données système 


A INSERT INTO [Proc 
| VALUES ('Con 
EH] INSERT INTO [Proc 
| VALUES ('Can 
EH INSERT INTO [Proc 
| VALUES ('Cui 


F INSERT INTO [Proc 
‘Ba 


E E] basecommerce 
& CA Schémas de base de données 
E Ca Tables 
Œ Tables système 


TA 


84 dbo.Rayc Nouvelle table... 
Æ © dbo.Rayc 

Œ Vues 

Œ Synonymes 

Œ Programmal 

Œ Service Brok 

Œ Sécurité 


Création 


Sélectionner les 1000 lignes du haut 


Modifier les 200 lignes du haut 


Générer un script de latableentantque + 


Afficher les dépendances 


FIGURE 37.35 — Afficher les premières lignes de la table 


Ce qui permet de voir la table Produits (voir la figure 37.36). 
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[HI Résutats | [Ja Messages | 

ID Nom Prix Stock Urilmage 
Î Télé HD 239 50 tele jpg 
A Console de jeux 150 25 console jpg 
3 Canapé 400 10 canape jpg 
4 Cuisinière 280 20 cuisiniere jpg 
5. Bouilloire 19 100 bouilloire jpg 
6 Li 2 places 149 15 lit jpg 
Z Pull 40 25 pull jpg 
8 T-shirt 20 20 tshirt jpg 
sg Pyjama 45 |4 pyiama jpg 
10 Tablette PC 350 44 tablette jpg 
11 Smartphone 320 40 smartphone jpg 


FIGURE 37.36 — Les premières lignes de la table Produits 


I n’y a plus qu’à relier les produits et les rayons. Pour cela, il faut relier les identifiants 
entre eux. Par exemple, avec le script suivant j’indique que le rayon Salon (identifiant 
1) contient la télé HD (identifiant 1), la console de jeux (identifiant 2), le canapé 
(identifiant 3), la tablette PC (identifiant 10). J’indique également que le rayon Cuisine 
(identifiant 2) contient la cuisinière (identifiant 4), ainsi que la bouilloire (identifiant 


5). Et ainsi de suite... : 


1| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
2 VALUES (1, 1) 

3[ INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
4 VALUES (1, 2) 

5|[ INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
6 VALUES (1, 3) 

7| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
8 VALUES (1, 10) 

9] INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
10 VALUES (2, 4) 

11| INSERT INTO [RayonProduit] ([Rayons_ID]l,[Produits_ID]) 
12 VALUES (2, 5) 

13 | INSERT INTO [RayonProduit] ([Rayons_ID]l,[Produits_ID]) 
14 VALUES (3, 3) 

15| INSERT INTO [RayonProduit] ([Rayons_ID]l,[Produits_ID]) 
16 VALUES (3, 6) 

17 | INSERT INTO [RayonProduit] ([Rayons_ID]l,[Produits_ID]) 
18 VALUES (3, 9) 

19|[ INSERT INTO [RayonProduit] ([Rayons_ID]l,[Produits_ID]) 
20 VALUES (4, 1) 

21| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
22 VALUES (4, 2) 

23| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
24 VALUES (4, 10) 

25| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
26 VALUES (4, 11) 

27| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
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28 VALUES (5, 7) 
29 | INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
30 VALUES (5, 8) 
31| INSERT INTO [RayonProduit] ([Rayons_ID],[Produits_ID]) 
32 VALUES (5, 9) 


Vous pouvez copier ce code grâce au code web suivant : 


Copier ce code ) 


Code web : 921140 


Voilà, notre base de données est prête. Nous allons pouvoir utiliser cette base et ses 
données dans notre code C;}! 


Se connecter à la base de données, lire et écrire 


Il est temps de se connecter à notre base de données depuis notre application C#. Pour 
cela, nous avons besoin de la chaîne de connexion à la base de données. Nous avons 
déjà parlé de la chaîne de connexion, elle contient toutes les informations nécessaires 
pour se connecter à la base de données ; à savoir le nom du serveur, le nom de la base, 
les identifiants de connexion, le type de connexion, etc. Nous en avons également vu un 
aperçu lorsque nous avons utilisé l’assistant de génération de modèle sauf que je vous 
ai indiqué que cette chaîne de connexion n’allait pas être bonne pour nos besoins. En 
effet, nous avons besoin qu’elle pointe vers notre serveur de base de données et pas vers 
le fichier temporaire que nous avons créé pour les besoins de l’assistant. Cette chaîne 
de connexion a toute sa place dans le fichier de configuration de l’application, que nous 
avons déjà étudié. Si vous ne l’avez pas déjà ajouté, il est temps de le faire. Et vous 
pouvez le remplir avec la configuration suivante : 


<?xml version="1.0" encoding="utf-8" ?7> 
<configuration> 
<connectionStrings?> 


& © NN 


<add name="NotreBaseDeDonnees" connectionString="metadata= 
res ://*x/ModelCommerce.csdl|res://*/ModelCommerce.ssdl| 
res ://*x/ModelCommerce.msl;provider=System.Data.SqlClient 
;provider connection string='data source=.\SQLEXPRESS ; 
Initial Catalog=basecommerce; integrated security=True'" 
providerName="System.Data.EntityClient" /> 

5 </connectionStrings> 

6| </configuration> 


Nous indiquons ici que notre chaîne de connexion va être accessible par le nom 
NotreBaseDeDonnees. Il y à plein d'informations dans l’attribut connectionString, 
mais ce qui nous intéresse surtout, c’est d'indiquer la source de données (à savoir : 
data source=.\SQLEXPRESS) ce qui va nous permettre d'indiquer que notre serveur 
est accessible à cette adresse, puis le nom de la base que nous avons créée (Initial 
Catalog-basecommerce) et enfin d'indiquer que nous utilisons l’authentification Win- 
dows (integrated security=True). Le reste permet de donner des informations de 
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description du modèle. Enfin, nous indiquons que nous utilisons les méthodes d’En- 
tity Framework pour l’accès aux données, à travers System.Data.EntityClient. Bref, 
beaucoup de ces informations sont issues de la chaîne de connexion générée par Visual 
C+# Express ; nous avons simplement changé le mode de connexion pour qu’il corres- 
ponde à nos besoins. Maintenant, nous allons pouvoir accéder à la chaîne de connexion 
avec le ConfigurationManager que nous connaissons bien désormais : 


1] string chaineConnexion = ConfigurationManager.ConnectionStrings 
['NotreBaseDeDonnees"].ConnectionString; 


Vous n’avez bien sûr pas oublié de référencer l’assembly System.Configuration! Re- 
tournons dans notre designer et cliquons dessus pour observer les propriétés du modèle. 
Nous allons modifier le nom du conteneur d’entités pour y mettre un nom un peu 
plus parlant, à savoir BaseDeDonnees (voir la figure 37.37). 


ModelCommerce ConceptualEntityModel % 


Accès au conteneur d'entités Public « 


Chaîne de connexion metadata=res://*/Model( 


Chargement différé activé True 

Espace de noms ModelCommerce 

Flux de génération de la base de do TablePerTypeStrategy.xan) 

Modèle de génération du DDL SSDLToSQL10.tt (VS) 
BaseDeDonnees 

Nom du schéma de la base de dont dbo 

Pluraliser les nouveaux objets False 


Nom du conteneur d'entités 


Nom du conteneur d'entités qui contient toutes les instances des 
entités du Entity Data Model. 


FIGURE 37.37 - Modification du nom du conteneur d’entités 


C’est le point d’entrée de notre accès aux données. Il s’agit en fait d’une classe qui a 
été générée par le designer d’Entity Framework. 


Une classe générée ? Où ça ? 


Dans l’explorateur de solutions, en dépliant le fichier ModelCommerce.edmx. Il s’agit 
d’un fichier intitulé ModelCommerce .Designer.cs qui contient la définition des classes 
générées et toute la logique permettant d’accéder à la base de données. Nous pou- 
vons l’ouvrir, mais le code est assez verbeux et nous risquons de nous perdre. Faisons 
confiance à Visual C;£ Express, nous allons utiliser son code généré les yeux fermés! 
Notez quand même que le code généré possède des classes préfixées par le mot-clé 
partial. Je vais y revenir plus loin. 


Vous ne vous en rendez peut-être pas encore compte, mais l’outil d’'ORM « Entity 
Framework » nous simplifie énormément la tâche (et je pèse mes mots, énormément !), 
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car non seulement il génère toutes les classes représentant les données en base (comme 
la classe Produit ou la classe Rayon) mais il s'occupe également de nous simplifier 
la création, la lecture ou la modification des données en base. Si nous avions dû le 
faire à la main comme c'était le cas avant l’utilisation d'ORMs, cela aurait mérité une 
centaine de pages supplémentaires d'explications et de code à comprendre. Là, nous 
nous positionnons en tant qu'utilisateur de ces classes générées et vous allez voir que 
c’est facile à utiliser ; vous n’imaginez pas le plaisir que c’est de constater que l'ORM 
a travaillé pour nous! 


Nous pouvons désormais instancier la classe BaseDeDonnees générée en lui passant en 
paramètre la chaîne de connexion : 


1] string chaineConnexion = ConfigurationManager .ConnectionStrings 
['NotreBaseDeDonnees"].ConnectionString; 

2| BaseDeDonnees baseDeDonnees = new BaseDeDonnees(chaineConnexion 
>; 


Vous voilà connectés à la base de données. Nous allons pouvoir utiliser les objets que 
Visual C# Express a générés à travers cette variable de type BaseDeDonnees, comme 
par exemple la propriété Rayons qui nous permet d'accéder aux rayons de notre base : 


1| foreach (Rayon rayon in baseDeDonnees.Rayons) 


2| € 

3 Console.WriteLine("{0} ({1})", rayon.Nom, rayon.Description 
); 

al} 


Ici, nous pouvons parcourir la liste des rayons avec un foreach car la propriété Rayons 
est du type ObjectSet<> qui implémente IEnumerable<>. 


Ce qui donne : 


Salon (Tout ce qu’on trouve dans un salon) 


Cuisine (Venez découvrir l’univers de la cuisine) 
Dormir () 

Hi-Tech (Les produits hi-tech 

Vêtements () 


De même, nous pouvons parcourir tous les produits grâce à la propriété Produits : 


1| foreach (Produit produit in baseDeDonnees.Produits) 

2| { 

3 Console.WriteLine("{0} : {1}", produit.Nom, produit.Prix); 
al} 


Ce qui donne : 


Télé HD : 299 

Console de jeux : 150 
Canapé : 400 
Cuisinière : 280 
Bouilloire : 19 
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Eit2 places 119 
BUS S TO 

TS ShEtE-70 
Pyjama : 15 
Tablette PC : 350 
Smartphone : 320 


Et ce qui est formidable, c’est qu’étant donné que la propriété Produits est énumé- 
rable, nous allons pouvoir y faire toutes les requêtes LINQ que nous le souhaitons, par 
exemple : 


1| IEnumerable <Produit> produits = from produit in baseDeDonnees. 
Produits 


2 where produit.Prix > 1650 

3 orderby produit.Prix descending 

4 select produit; 

5| foreach (Produit produit in produits) 

6| 

7 Console.WriteLine("{O} : {1}", produit.Nom, produit.Prix); 
8|} 


Nous obtenons tous les produits dont le prix est supérieur à 150, triés par prix décrois- 
sant : 


Canapé : 400 
Tablette PC : 350 


Smartphone : 320 
Télé HD : 299 
Cuisinière : 280 


Pratique! Tout le SQL nécessaire pour renvoyer cette liste de produits filtrée a été 
généré par Entity Framework. Nous n’avons rien à faire d’autre que d'utiliser le C#. 


Et voilà. Avouez que c’est quand même super simple, non ? Avouez également que, si 
vous avez l'habitude de tout faire à la main dans un autre langage de programmation, 
vous êtes émerveillés! J’exagère peut-être un peu, mais Entity Framework nous fait 
gagner un temps considérable au développement ainsi que tout au long de la vie de 
l'application. 


Merci à lui de nous avoir généré tout le code adéquat. À propos de génération de code, 
souvenez-vous que les classes générées sont partielles. Nous en avons déjà parlé dans le 
chapitre dédié, mais je vous rappelle le but ici. I s’agit de permettre d’ajouter des fonc- 
tionnalités à la classe sans avoir à modifier le fichier ModelCommerce .Designer.cs. En 
effet, à chaque fois que nous faisons une modification sur notre modèle (ajout d’entité, 
changement de nom, etc.), il régénère toutes les classes de ce fichier. Si nous avions 
modifié des choses à la main dedans, elles vont disparaître... Le mot-clé partial nous 
offre l’opportunité d'ajouter des fonctionnalités à la classe depuis un autre fichier. Nous 
pouvons en profiter pour rajouter nos propres méthodes, par exemple une méthode qui 
renvoie les produits dont le prix est supérieur à un prix passé en paramètre. Il suffit de 
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déclarer une classe partielle du même nom que la classe BaseDeDonnees, située dans le 
même espace de noms et de rajouter des méthodes. Par exemple : 


1| public partial class BaseDeDonnees 


2| € 

3 public IEnumerable <Produit> ProduitsPlusCherQue(decimal 
prix) 

4 { 

5 return from produit in Produits 

6 where produit.Prix > prix 

7 select produit; 

8 1 

o[ } 


Nous pourrons donc utiliser cette méthode de cette façon : 


1| foreach (Produit produit in baseDeDonnees.ProduitsPlusCherQue( 
200)) 
2| { 
3 Console.WriteLine("{0} : {1}", produit.Nom, produit.Prix); 
} 


Ce qui donnera : 


Teen n299 
Canapé : 400 


Cuisinière : 280 
Tablette PC : 350 
Smartphone : 320 


Remarquons que chaque Rayon possède également une propriété Produits; c’est la 
propriété de navigation que nous avons renommée précédemment. Entity Framework 
a donc compris qu’il y avait une relation entre les rayons et les produits et il permet 
d'accéder aux produits qui font partie du rayon, grâce à cette propriété. Aïnsi, nous 
pouvons écrire un code, comme le suivant, qui accède à la propriété Produits d’un 
rayon et permet d’afficher la liste de tous les produits de chaque rayon : 


1| foreach (Rayon rayon in baseDeDonnees.Rayons) 


2| € 

3 Console.WriteLine("{0} ({1})", rayon.Nom, rayon.Description 
); 

4 foreach (Produit produit in rayon.Produits) 

5 { 

6 Console.WriteLine("\t{O} : {1}", produit.Nom, produit. 

Prix); 
7 } 
8| } 


Sauf qu'ici nous rencontrons un problème. Si nous exécutons ce bout de code, nous 
aurons l’exception suivante : 
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Exception non gérée : System.Data. 
EntityCommandExecutionException: Une erreur s’est produite 
lors de l’exécution de la définition de la commande. Pour 


plus de détails, consultez l’exception interne. ---> System. 
InvalidOperationException: Un DataReader associé à cette 
Command est déjà ouvert. IL doit d’abord être fermé. 


1 


Pourquoi un tel problème ? 


En fait, cela vient de la façon dont sont récupérées les données. Lorsque nous accédons 
à la propriété Rayons, Entity Framework génère une requête en base de données pour 
récupérer la liste des rayons. Puis à l’intérieur de la boucle, lorsque nous accédons à 
la propriété Produits d’un rayon, il génère à nouveau une requête pour récupérer les 
produits de ce rayon. Il y a donc deux connexions à la base de données en même temps, 
et Ça, il ne sait pas le faire par défaut. 


Il y a plusieurs façons de corriger le problème. La première est de changer la chaîne 
de connexion en rajoutant une directive permettant de préciser qu’on autorise l’accès 
multiple, à savoir : 


1 | multipleactiveresultsets=True; 


La chaîne de connexion devient donc : 


1] <add name="NotreBaseDeDonnees'" connectionString="metadata=res 
://*/ModelCommerce.csdllres ://*/ModelCommerce.ssdllres://x*/ 
ModelCommerce.msl;provider=System.Data.SqlClient ; provider 
connection string='data source=.\SQLEXPRESS ; Initial Catalog= 
basecommerce; integrated security=True; 
multipleactiveresultsets=True;'" providerName="System.Data. 
EntityClient" /> 


Avec ce changement, si nous exécutons ce code, nous aurons : 


Salon (Tout ce qu’on trouve dans un salon) 
Télé HD : 299 
Console de jeux : 150 
Canapé : 400 
Tablette PC : 350 
Cuisine (Venez découvrir l’univers de la cuisine) 
Cuisinière : 280 
Bouilloïre : 19 
Dormir () 
Canapé : 400 
FE2S places MX) 
Pyjama : 15 
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Hi-Tech (Les produits hi-tech ...) 
Télé HD : 299 
Console de jeux : 150 
Tablette PC : 350 
Smartphone : 320 

Vêtements () 
QUI TO 
TS 20 
Pyjama : 15 


La deuxième solution est de faire en sorte que la première requête soit terminée avant 
l’exécution des suivantes. Pour cela, il suffit de forcer l'évaluation de la requête en 
utilisant par exemple un ToList() : 


1| foreach (Rayon rayon in baseDeDonnees.Rayons.ToList ()) 


2! { 


3 Console.WriteLine("{0} ({1})", rayon.Nom, rayon.Description 
); 

4 foreach (Produit produit in rayon.Produits) 

5 À 

6 Console.WriteLine("\t{O} : {1}", produit.Nom, produit. 

Prix); 
7 } 
8| } 


Ceci est possible, car Entity Framework bénéficie de l’exécution différée ; le ToList() 
résout le problème en forçant l'exécution de la requête. 


Enfin, la dernière solution est de faire en sorte que la première requête qui charge les 
Rayons inclue également le chargement des produits. Ainsi, il n’y a qu’une seule et 
unique requête qui charge tout. Cela se passe avec la méthode Include, en précisant 
le nom de la propriété de navigation à charger : 


1| foreach (Rayon rayon in baseDeDonnees.Rayons.Include ("Produits" 


)) 
2| € 
3 Console.WriteLine("{0} ({1})", rayon.Nom, rayon.Description 
); 
4 foreach (Produit produit in rayon.Produits) 
5 { 
6 Console.WriteLine("\t{O} : {1}", produit.Nom, produit. 
Prix); 
dé } 
8| } 


Il y a plusieurs choses à remarquer ici. 


Nous avons vu qu’Entity Framework est capable d’aller chercher les données liées entre 
elles grâce aux propriétés de navigation. Les deux premiers scénarios couverts tirent 
parti de ce qu’on appelle le {azy loading que l’on peut traduire en « chargement pares- 
seux ». Cela veut dire que c’est uniquement lorsque l’on accède à la propriété Produits 


459 


CHAPITRE 37. ACCÉDER AUX DONNÉES AVEC ENTITY FRAMEWORK 


qu'Entity Framework va aller lire le contenu associé en base de données. Ceci implique 
qu’à chaque tentative d’accès à la propriété Produits d’un rayon, Entity Framework 
va effectuer une requête en base de données pour ramener les produits concernés. C’est 
très bien si on fait ça une ou deux fois, mais dans notre cas au final on fait autant de 
requête qu’il y a de rayons. Ce qui n’est pas très performant... Le troisième scénario 
montre l’utilisation de la méthode Include qui permet de tout rapatrier en une seule 
requête, ce qui est évidemment plus performant. 


Faites bien attention à votre utilisation d'Entity Framework. Si ce n'est pas 
très grave pour une petite application, cela peut le devenir pour des applica- 
tions avec des grosses bases de données. 


Alors, vous ne trouvez pas que la lecture en base de données est particulièrement aisée ? 
Merci Entity Framework! Notons que nous pouvons également accéder aux rayons 
dans lesquels sont positionnés les produits grâce à la propriété Rayons. Nous pourrions 
éventuellement nous en servir pour afficher le nombre de rayons dans lesquels le produit 
est présent. 


L'écriture en base de données est tout aussi aisée. Le principe est d’ajouter des valeurs 
à notre objet de base de données et de sauvegarder les modifications. 


Pour ajouter un nouveau rayon, il suffit d'appeler la méthode AddObject disponible 
sur la propriété Rayons. Il ne faudra pas oublier d'appeler la méthode SaveChanges qui 
s'occupe d'insérer physiquement les valeurs en base de données. Pour créer un nouveau 
rayon, il suffira d’instancier un objet Rayon, de renseigner des propriétés, de créer des 
produits et de les ajouter au rayon, par exemple : 


1| Rayon rayon = new Rayon(); 

2| rayon.Nom = "Vins"; 

3| rayon.Description = "Venez découvrir notre sélection des plus 
grands châteaux"; 

4 

5| Produit produiti = new Produit (); 

6| produiti.Nom = "Château ronto'"; 

7| produiti.Prix = 9.99M; 

8| produiti.Stock = 60; 

9| produiti.UrlIimage = "à compléter ..."; 

10 

11| Produit produit2 = new Produit (); 

12] produit2.Nom = "Château toro'; 


13| produit2.Prix = 15; 

14| produit2.Stock = 6; 

15| produit2.Urlimage = "à compléter ..."; 
16 

17 | rayon.Produits.Add(produit1); 

18| rayon.Produits.Add(produit2); 

19 

20 | baseDeDonnees .Rayons.AddObject (rayon); 
21| baseDeDonnees.SaveChanges(); 
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Ainsi, si nous réaffichons la liste des rayons, nous pourrons voir un rayon de plus 
contenant des produits en plus... 


Salon (Tout ce qu’on trouve dans un salon) 
Télé HD : 299 
Console de jeux : 150 
Canapé : 400 
Tabl'ette_PCu_:_350 
Cuisine (Venez découvrir l’univers de la cuisine) 
Cuisinière : 280 
Bouilloire; 19) 
Dormir () 
Canapé : 400 
Pit 2,pl'aces 149 
Pyjama : 15 
(Les produits hi-tech 
HEILS Ein) à 200) 
Console de jeux : 150 
Tablette PC : 350 
Smartphone : 320 
Vêtements () 
Pull : 40 
Tshirt 20 
Pyjama : 15 
Vins (Venez découvrir notre sélection des plus grands châteaux) 
Château ronto : 10 
Château toro : 15 


De même, si vous allez voir en base de données, vous aurez bien les nouveaux éléments, 
ainsi que l’illustre la figure 37.38. 


On observe la création d’un nouveau rayon, de deux nouveaux produits et nous avons 
bien dans la table de relation les nouveaux produits reliés au nouveau rayon. Il est 
également possible d'ajouter un produit à un rayon. Nous pouvons le faire de deux 
manières différentes. La première est d’ajouter un Produit directement dans la collec- 
tion Produits d’un rayon, Il sera directement ajouté dans le rayon de notre choix. La 
deuxième est d'ajouter un produit dans la collection Produits et si nous voulons qu’il 
soit présent dans un rayon, il faudra que sa propriété Rayons contienne les rayons dans 
lesquels nous souhaitons ajouter le produit. Voyons la première méthode : 


Rayon rayon = baseDeDonnees.Rayons.First(r => r.Nom == "Vins"); 


Produit produit = new Produit (); 
produit.Nom = "Chateau pinière'; 
produit.Prix = 12.50M; 
produit.Stock = 40; 

produit .UrlImage = "vin.jpg"; 
produit.Rayons.Add(rayon); 


© OO I OO Où À À D 
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baseDeDonnees.SaveChanges (); 
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TH Résutats | [3 Messages | 
1 i 1 j Salon Tout ce qu'on trouve dans un salon 


2 2 Cuisine Venez découvrir l'univers de la cuisine 
3 3  Domir NULL 
4 4 Hi-Tech Les produits hitech … 


9 9 Pyjama 15 |4 pyjama jpg 
10 10 Tablette PC 350 44 tablette jpg 
11 11 Smartphone 320 40 smartpho… 
Chêteau ronto 
Chêteau toro 


FIGURE 37.38 — Le nouveau rayon est les nouveaux produits sont visibles dans les tables 


Nous commençons par récupérer un rayon, puis nous instancions un objet de type 
Produit. Enfin, nous faisons le lien entre le produit et le rayon en ajoutant le rayon à la 
collection Rayons de notre objet produit. Comme d'habitude, la méthode SaveChanges () 
permet de faire persister les informations. 


La seconde méthode est un peu plus simple à appréhender ; il suffit d’instancier un 
objet Produit et de l’ajouter à la collection Produits d’un rayon : 


Rayon rayon = baseDeDonnees.Rayons.First(r => r.Nom == "Vins"); 


1 

2 

3| Produit produit = new Produit (); 
4| produit.Nom = "Chateau réro'; 

5| produit .Prix = 3.20M; 

6| produit.Stock = 10; 

7| produit .UrlImage = "vini.jpg"; 

8 
9 


rayon.Produits.Add(produit); 
10 
11] baseDeDonnees.SaveChanges() ; 


Dans les deux cas, Entity Framework arrive à faire le lien entre un rayon et un produit. 
Le château pinière et le château réro ont bien été ajoutés...! Vous pouvez également 
vérifier en base de données que la relation entre le rayon et le produit a bien été faite. 


Il est également possible de modifier des enregistrements en base de données. Le prin- 
cipe est de modifier l’élément concerné dans l’objet de base de données et d’appeler la 
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méthode SaveChanges() : 


1| Rayon rayon = baseDeDonnees.Rayons.First(r => r.Nom == "Vins"); 
2| rayon.Description = "Les meilleurs vins"; 
3| baseDeDonnees.SaveChanges(); 


Enfin, nous pouvons aussi supprimer des données en base. Le principe est le même 
que pour l’ajout. Nous appelons une méthode qui s’occupe de la suppression et nous 
appelons la méthode SaveChanges. Par contre, il faut faire attention à l’intégrité des 
données. On ne peut pas supprimer un rayon qui a des produits dedans. Il faut com- 
mencer par retirer la relation entre les produits et le rayon : 


Rayon rayon = baseDeDonnees.Rayons.First(r => r.Nom == "Vins"); 
rayon.Produits.Clear(); 

baseDeDonnees.DeleteUbject (rayon); 

baseDeDonnees.SaveChanges (); 


& & ND OH 


Le fait d'appeler la méthode Clear() sur les produits du rayon vide le rayon de ses 
produits. Il n’y a donc plus de produits dans ce rayon, mais ils existent toujours en 
base de données car ils n’ont pas été supprimés physiquement. C’est important car 
ces produits peuvent également être présents dans d’autres rayons, nous ne pouvons 
donc pas les supprimer. Ensuite, on utilise la méthode DeleteUbject pour supprimer 
un élément de la base de données. Ici, nous supprimons le rayon et nous validons les 
modifications avec SaveChanges(). 


Notez que si nous n’avions pas vidé le rayon de ses produits, la suppression du rayon 
aurait été impossible car comme il existe une relation entre les produits et les rayons 
et que notre base de données possède une contrainte d’intégrité (la clé étrangère) entre 
les produits et le rayon, la suppression aurait provoqué une erreur. En effet, la table 
RayonProduit contiendrait un identifiant de rayon qui n’existe plus. C’est impossible! 
La contrainte de la clé étrangère est là pour nous assurer que nous ne faisons pas 
n'importe quoi dans la base de données et qu’elle est toujours cohérente. Si nous l’avions 
fait, nous aurions eu l’exception suivante : 


Exception non gérée : System.Data.UpdateException: Une erreur s? 
est produite lors de la mise à jour des entrées. Pour plus d’? 
informations, consultez l’exception interne. ---> System.Data 
.SqlClient.SqlException: L’instruction DELETE est en conflit 


avec la contrainte REFERENCE "FK\_RayonProduit\_Rayon". Le 
conflit s’est produit dans la base de données "basecommerce", 
table "dbo.RayonProduit", column ?’Rayons\_ID”?. 

L’instruction a été arrêtée. [...] 


Remarquez que nous aurons désormais des produits orphelins. Tout dépend de ce que 
l’on veut faire maintenant. Souhaitons-nous qu'ils soient supprimés également vu qu'ils 
ne sont plus dans aucun rayon ? Souhaitons-nous qu'ils restent présents pour pouvoir 
les ajouter ultérieurement à un autre rayon ? Ça, c’est vous qui décidez! Maintenant 
que vous savez supprimer des objets, vous pouvez faire comme bon vous semble. 


Notons avant de terminer qu’il est tout à fait possible de faire plusieurs ajouts, modi- 
fications ou suppressions en même temps. Il suffira de terminer toutes les instructions 
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par la méthode SaveChanges() qui s’arrangera pour tout faire persister ! 


En résumé 


— Entity Framework est l’outil de mapping objet relationnel de Microsoft permettant 
de travailler sur une base de données relationnelle avec une approche orientée objet. 

— Entity Framework est capable de modéliser des données et de générer les tables 
correspondantes en base de données sans qu’il soit nécessaire de maîtriser le SQL. 

— Il simplifie grandement la lecture et l'écriture des données en base et tire parti, si 
besoin, des mécanismes de chargement paresseux. 
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Les tests unitaires 


Difficulté : (3 


ne des grandes préoccupations des créateurs de logiciels est d'être certains que 

leur application informatique fonctionne et surtout qu'elle fonctionne dans toutes les 

situations possibles. Nous avons tous déjà vu notre système d'exploitation planter, 
ou bien notre logiciel de traitement de texte nous faire perdre les 50 pages de rapport que 
nous étions en train de taper. Ou encore, un élément inattendu dans un jeu où l'on arrive 
à passer à travers un mur alors qu'on ne devrait pas... 


Bref, pour être sûr que son application fonctionne, il faut faire des tests. 
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Qu'est-ce qu’un test unitaire et pourquoi en faire ? 


Un test constitue une façon de vérifier qu’un système informatique fonctionne. 


Tester son application c’est bien. Il faut absolument le faire. C’est en général une pra- 
tique plutôt laissée de côté, car rébarbative. Il y à plusieurs façons de faire des tests. 
Celle qui semble la plus naturelle est celle qui se fait manuellement. On lance son ap- 
plication, on clique partout, on regarde si elle fonctionne. Celle que je vais présenter 
ici constitue une pratique automatisée visant à s’assurer que des bouts de code fonc- 
tionnent comme il faut et que tous les scénarios d’un développement sont couverts par 
un test. Lorsque les tests couvrent tous les scénarios d’un code, nous pouvons assurer 
que notre code fonctionne. De plus, cela permet de faire des opérations de maintenance 
sur le code tout en étant certain qu’il n’aura pas subi de régressions. De la même façon, 
les tests sont un filet de sécurité lorsqu'on souhaite refactoriser son code ou l’optimiser. 
Cela permet dans certains cas d’avoir un guide pendant le développement, notamment 
lorsqu'on pratique le TDD. Le Test Driven Development (TDD) ! est une méthode de 
développement de logiciel qui préconise d’écrire les tests unitaires avant d’écrire le code 
source d’un logiciel. Nous y reviendrons ultérieurement. 


Q Un test est donc un bout de code qui permet de tester un autre code. 


En général, un test se décompose en trois parties, suivant le schéma « AAA », qui 
correspond aux mots anglais Arrange, Act, Assert, que l’on peut traduire en français 
par : Arranger, Agir, Auditer. 


— Arranger : il s’agit dans un premier temps de définir les objets, les variables néces- 
saires au bon fonctionnement de son test (initialiser les variables, initialiser les objets 
à passer en paramètres de la méthode à tester, etc.). 

— Agir : ensuite, il s’agit d'exécuter l’action que l’on souhaite tester (en général, exé- 
cuter la méthode que l’on veut tester, etc.). 

— Auditer : enfin, il faut vérifier que le résultat obtenu est conforme à nos attentes. 


Notre premier test 


Imaginons que nous voulions tester une méthode toute simple qui fait l'addition entre 
deux nombres, par exemple la méthode suivante : 


1| public static int Addition(int a, int b) 
2| { 

3 return à + b; 

al} 


Faire un test consiste à écrire des bouts de code permettant de s'assurer que le code 
fonctionne. Cela peut-être par exemple : 


1. En français, le TDD se dit « développement piloté par les tests ». 
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1| static void Main(string[] args) 

2! { 

3 // arranger 

4 int a = 1; 

5 int b = 2; 

6 // agir 

7 int resultat = Addition(a, b); 

8 // auditer 

9 if (resultat != 3) 

10 Console.WriteLine("Le test a raté"); 


11| } 


Ici, le test passe bien, ouf! Pour être complet, le test doit couvrir un maximum de 
situations ; il faut donc tester notre code avec d’autres valeurs, et ne pas oublier les 
valeurs limites : 


1| static void Main(string[] args) 

2| À 

8 int a = 1; 

4 int b = 2; 

5 int resultat = Addition(a, b); 

6 if (resultat != 3) 

7 Console.WriteLine("Le test a raté"); 
8 a = O0; 

9 b = O0; 

10 resultat = Addition(a, b); 

11 if (resultat != 0) 

12 Console.WriteLine("Le test a raté"); 
13 a = -B5; 

14 b = 5; 

15 resultat = Addition(a, b); 

16 if (resultat != 0) 

17 Console.WriteLine("Le test a raté"); 
181 } 


Voilà pour le principe. Ici, nous considérons avoir écrit suffisamment de tests pour 
nous assurer que cette méthode est bien fonctionnelle. Bien sûr, cette méthode était 
par définition fonctionnelle, mais il est important de prendre le réflexe de tester des 
fonctionnalités qui sont déterminantes pour notre application. 


Voyons maintenant comment nous pourrions tester une méthode avec l'approche TDD. 
Pour rappel, lors d’une approche TDD), le but est de pouvoir faire un développement à 
partir des cas de tests préalablement établis par la personne qui exprime le besoin ou 
suivant les spécifications fonctionnelles. 


Imaginons que nous voulions tester une méthode qui calcule la factorielle d’un nombre. 
Nous savons que la factorielle de 0 vaut 1, la factorielle de 1 vaut 1. Commençons par 
écrire les tests : 

1| static void Main(string[] args) 


2| À 


à int valeur = 0; 
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4 int resultat = Factorielle(valeur); 

5 if (resultat != 1) 

6 Console.WriteLine("Le test a raté"); 
7 

8 valeur = 1; 

9 resultat = Factorielle(valeur); 

10 if (resultat != 1) 

11 Console.WriteLine("Le test a raté"); 
12| } 


Le code ne compile pas ! Forcément, nous n’avons pas encore créé la méthode Factorielle. 
C’est la première étape. La suite de la méthode est de faire en sorte que le test compile, 
mais il échouera puisque la méthode n’est pas encore implémentée : 


1| public static int Factorielle(int a) 

2 |: € 

3 throw new NotImplementedException(); 
al} 


Il faudra ensuite écrire le code minimal qui servira à faire passer nos deux tests. Cela 
peut-être : 


1| public static int Factorielle(int a) 
2| { 

3 return 1; 

al} 


Si nous exécutons nos tests, nous voyons que cette méthode est fonctionnelle car ils 
passent tous. La suite de la méthode consiste à refactoriser le code, à l’optimiser. Ici, 
il n’y a rien à faire, car c’est vraiment simple. On se rend compte par contre qu’on 
n’a pas couvert énormément de cas de tests ; faire juste des tests avec 0 et 1 c’est un 
peu léger... Nous savons que la factorielle de 2 vaut 2, la factorielle de 3 vaut 6, la 
factorielle de 4 vaut 24, etc. Continuons à écrire des tests (il faut bien sûr garder les 
anciens tests afin d’être sûrs qu’on couvre un maximum de cas) : 


1| static void Main(string[] args) 

2| À 

à int valeur = 0; 

4 int resultat = Factorielle(valeur); 

5 if (resultat != 1) 

6 Console.WriteLine("Le test a raté"); 
7 

8 valeur = 1; 

9 resultat = Factorielle(valeur); 

10 if (resultat != 1) 

11 Console.WriteLine("Le test a raté"); 
12 

13 valeur = 2; 

14 resultat = Factorielle(valeur); 

15 if (resultat != 2) 

16 Console.WriteLine("Le test a raté"); 
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17 
18 
19 
20 
21 
22 
23 
24 
25 
26 
27 


} 


valeur = 3; 
resultat = Factorielle(valeur); 
if (resultat != 6) 


Console.WriteLine("Le test a raté"); 


valeur = 4; 
resultat = Factorielle(valeur); 
if (resultat != 24) 


Console.WriteLine("Le test a raté"); 


Et nous pouvons écrire une méthode Factorielle qui fait passer ces tests : 


1 
2 
3 
4 
5 
6 
wi 
8 
9 


10 


public static int Factorielle(int a) 


{ 


} 


if (a == 2) 
return 2; 
if (a == 
return 6; 
if (a == 4) 
return 24; 
return 1; 


Lançons les tests : nous voyons que tout est Ok. Cependant, nous n’allons pas faire des 
if en déclinant tous les cas possibles, il faut donc repasser par l’étape d'amélioration 
et de refactorisation du code, afin d'éviter les redondances de code, d'améliorer les 
algorithmes, etc. Cette opération devient sans risque puisque le test est là pour nous 
assurer que la modification que l’on vient de faire est sans régression, si le test passe 
toujours bien sûr... Nous voyons que nous pouvons améliorer le code en utilisant la 
vraie formule de la factorielle : 


1 
2 
3 
4 
5 
6 
7 
8 
9 


public static int Factorielle(int a) 


{ 


} 


int total = 1: 
for (int i = 1 ; i <= a ; i ++) 
À 
total *= ji; 
} 


return total; 


Ce qui permet d'illustrer que, par exemple, la factorielle de 5 est égale à 1xX2X3Xx4X5. 
Relançons nos tests, ils passent tous. Parfait. Nous sommes donc certains que notre 
changement de code n’a pas altéré la fonctionnalité, car les tests continuent de passer. 
On peut même rajouter des tests pour le plaisir, comme la factorielle de 10, histoire 
d’avoir quelque chose d’un peu plus grand : 


1 
2 


valeur = 10; 


resultat = Factorielle(valeur); 
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3| if (resultat !-= 3628800) 
4 Console.WriteLine("Le test a raté"); 


Est-ce que cette méthode est optimisable ? Sûrement. Est-ce qu’il y a un risque à opti- 
miser cette méthode ? Aucun! En effet, nos tests nous garantissent que s’ils continuent 
à passer, alors une optimisation n’entraîne pas de régression dans la fonctionnalité. On 
sait par exemple qu’il y a un autre moyen pour calculer une factorielle. Par exemple, 
pour calculer la factorielle de 5, il suffit de multiplier 5 par la factorielle de 4. Pour 
calculer la factorielle de 4, il faut multiplier 4 par la factorielle de 3, et ainsi de suite 
jusqu’à arriver à 1... Bref, pour obtenir une factorielle on peut se servir du résultat de 
la factorielle du nombre précédent. Ce qui peut s’écrire : 


1| public static int Factorielle(int a) 
21 { 

3 if (a <= 1) 

4 return ll; 

5 return a * Factorielle(a - 1); 

6| } 


Ici la méthode Factorielle est une méthode récursive, c’est-à-dire qu’elle s’appelle 
elle-même. Cela nous permet d’indiquer que la factorielle d’un nombre correspond à ce 
nombre multiplié par la factorielle du nombre précédent. Bien sûr, il faut s’arrêter à un 
moment dans la récursion. On s’arrête ici quand on atteint le chiffre 1. Pour s’assurer 
que cette factorielle fonctionne bien, il suffit de relancer les tests. Tout est Ok, c’est 
parfait ! 


Voilà donc un exemple de TDD. Bien sûr, la méthode est ici poussée au maximum 
pour que vous compreniez l'intérêt de cette pratique. On peut gagner du temps en 
partant directement sur la bonne implémentation. Vous verrez qu’il y a toujours des 
premiers essais qui satisfont les tests mais qu'il sera possible d'améliorer régulièrement 
le code. Ceci devient possible grâce aux tests qui nous assurent que tout continue à bien 
fonctionner. La pratique du TDD dépend de la façon dont le développeur appréhende 
sa philosophie de développement. Elle est présentée ici pour sensibiliser ce dernier à 
cette pratique mais son utilisation n'est pas du tout obligatoire. Voilà pour les tests 
basiques. Cependant, utiliser une application console pour faire ses tests, ce n’est pas 
très pratique, vous en conviendrez. Nous avons besoin d'outils! 


Le framework de test 


Un framework de test est aux tests ce qu’un IDE est au développement. Il fournit un 
environnement structuré permettant l’exécution de tests, ainsi que des méthodes pour 
aider au développement de ceux-ci. Il existe plusieurs frameworks de test. Microsoft 
dispose de son framework, mstest, qui est disponible dans les versions payantes de 
Visual Studio. Son intérêt est qu'il est fortement intégré à l’'IDE. Son défaut est qu’il 
ne fonctionne pas avec les versions gratuites de l’environnement de développement. 
Comme nous sommes partis dans ce livre avec la version gratuite, Visual C7 Express, 
nous n’allons pas pouvoir utiliser mstest. 
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Par contre, il existe d’autres framework de test, gratuits, comme le très connu NUnit. 
NUnit est la version .NET du framework XUnit, qui se décline pour plusieurs envi- 
ronnements, avec par exemple PHPUnit pour le langage PHP, JUnit, pour java, etc. 
Première chose à faire : installer NUnit. Pour cela, utilisez le code web suivant : 


[T élécharger NUnit 
Code web : 481629 


La version que j'utilise dans ce livre est la version 2.5.10. Démarrez l'installation, comme 
indiqué à la figure 38.1. 


Welcome to the NUnit 2.5.10 Setup 
Wizard 


The Setup Wizard will install NUnit 2.5. 10 on your computer. 
Click Next to continue or Cancel to exit the Setup Wizard. 


FIGURE 38.1 -— Installation de NUnit 


L'installation est en anglais, mais assez facile à suivre. Cliquez sur Next puis, après 
avoir accepté la licence, vous pouvez choisir l'installation classique (voir la figure 38.2). 


Une fois le framework de test installé, nous pouvons créer un nouveau projet qui 
contiendra une fonctionnalité à tester. Je l'appelle MaBibliothequeATester. En gé- 
néral, nous allons surtout tester des assemblys avec NUnit. Je crée donc un projet de 
type bibliothèque de classes. Ce projet ne sera pas exécutable, car il ne s’agit pas d’une 
application console. À l’intérieur, je vais pouvoir créer une classe utilitaire, disons Math, 
qui contiendra notre méthode de calcul de factorielle : 


1| public static class Math 

2| { 

3 public static int Factorielle(int a) 
4 { 

5 if (a <= 1) 

6 réturn 1; 

7 return a * Factorielle(a - 1); 

8 } 

9! } 


A71 


CHAPITRE 38. LES TESTS UNITAIRES 


4] NUnit 2.5.10 Setup | 0 Es) 
Choose Setup Type 
Choose the setup type that best suits your needs (Que) 
ostalethomeste on program features, Recommended for most users. 


Allows users to choose which program features will be installed and where 
they will be installed. Recommended for advanced users, 


Complete 
All program features will be installed. Requires the most disk space. 


FIGURE 38.2 — Choix de l'installation classique 


Ensuite, ajoutons un nouveau projet de type bibliothèque de classes où nous allons 
mettre nos tests unitaires, appelons-le MathTests.Unit. Ce n’est pas une norme abso- 
lue, mais je vous conseille de suffixer vos projets de test avec .Unit, afin de les identifier 
facilement. 


Les tests doivent se mettre dans une classe spéciale. Ici aussi, pas de règle de nommage 
obligatoire, mais il est intéressant d’avoir une norme pour s’y retrouver facilement. Je 
vous propose de nommer les classes de tests en commençant par le nom de la classe que 
l’on doit tester, suivie du mot Tests. Ce qui donne : MathTests. Pour être reconnue 
par le framework de test, la classe doit respecter un certain nombre de contraintes. Elle 
doit dans un premier temps être décorée de l’attribut [TestFixture]. Il s’agit d’un 
attribut qui permet à NUnit de reconnaître les classes qui contiennent des tests. 


Cet attribut étant dans une assembly de NUnit, vous devez rajouter une référence à 
l’assembly nunit . framework, ainsi qu'illustré à la figure 38.8. 


Vous devez ensuite inclure l’espace de noms adéquat : 


1 | using NUnit.Framework; 


Nous allons pouvoir créer des méthodes à l’intérieur de cette classe. De la même façon, 
une méthode pourra être reconnue comme une méthode de test si elle est décorée de 
l’attribut [Test]. Ici aussi, il est intéressant de suivre une règle de nommage afin de 
pouvoir identifier rapidement l'intention de la méthode de test. Je vous propose le 
nommage suivant : MethodeTestee_Etatlnitial_EtatAttendu(). Par exemple, une 
méthode de test permettant de tester la factorielle pourrait s'appeler : 


[TestFixture] 
public class MathTests 
{ 
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c] es 
NET [COM | Projets | Parcourir | Récent 

Filtré pour : .NET Framework 4 
Nom du composant | Version Runtime Chemin d'acces # 
Microsoft.Windows.Design  3.5.0.0 v2.0,50727 c\Program Files\Microso. 
Microsoft.Windows.Desig...  3.5.0.0 v2.0,50727 c\Program Files\Microso. 
Microsoft.Windows.Desig...  4.0.0.0 v2.0,50727 C:\Program Files\Microso.. } 
Microsoft. Windows.Desig..  3.5.0.0 v2.0,50727 c:\Program Files\Microso.. 
Microsoft.Windows.Desig...  4.0.0.0 v2.0,50727 C:\Program Files\Microso.. 
Microsoft. Windows.Desig.  3.5.0.0 v2.0.50727 c:\Program Files\Microso. 
mscorlib 40.00 v4.0.30319 C:\Program Files\Referen.. 

| nunit.framework 251011092  v2.0.50727 C:\Program Files\NUnit 2... 
nunit.mocks 251011092  v2.0.50727 C:\Program Files\NUnit 2... 
PresentationBuildTasks 40.00 v4.0.30319 C:\Program Files\Referen.. 
PresentationCore 40.00 v4.0.30319 C:\Program Files\Referen… + 
4 il J » 


FIGURE 38.3 — Ajout d’une référence au framework de test 


5 public void Factorielle_AvecValeur3_Retourne6() 
6 { 

7 // test à faire 

8 , 

9 


} 


Il existe plein d’autres attributs que vous découvrirez ultérieurement. Il est temps de 
passer à l'écriture du test et surtout à la vérification du résultat. Pour cela, on utilise 
des méthodes de NUnit qui nous permettent de vérifier par exemple qu’une valeur est 
égale à une autre attendue. Cela se fait grâce à la méthode Assert.AreEqual () : 


1] [Test] 

2| public void Factorielle_AvecValeur3_Retourne6() 

31 À 

4 int valeur = 3; 

5 int resultat = MaBibliothequeATester.Math.Factorielle( 
valeur); 

6 Assert.AreEqual(6, resultat); 

7} 


Elle permet de vérifier que la variable valeur vaut bien 6. Rajoutons tant qu’on y est 
une méthode de test qui échoue : 


1] [Test] 

2| public void Factorielle_AvecValeur10_Retournel () 

31 À 

4 int valeur = 10; 

5 int resultat = MaBibliothequeATester.Math.Factorielle( 


valeur); 
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6 Assert.AreEqual(i, resultat, "La valeur doit être égale à 1 


hs 


Il'est important que chaque méthode qui s'occupe de tester une fonctionnalité, 
le fasse à l'aide d'un cas unique comme illustré juste au-dessus. La première 
méthode teste la fonctionnalité Factorielle pour le cas où la valeur vaut 
3 et la seconde s'occupe du cas où la valeur vaut 10. Vous pouvez rajouter 
autant de méthodes de tests que vous le souhaitez tant qu'elles sont décorées 


de l'attribut [Test]. 


J’en ai profité pour ajouter un message qui permettra d'indiquer des informations com- 
plémentaires si le test échoue. Compilez maintenant le projet, allez dans le répertoire 
d'installation de NUnit (C:\Program Files\NUnit 2.5.10\bin\net-2.0) et lancez 
l'application nunit .exe (voir figure 38.4). 


EF 


É 


View Test Tools Help 


{ Categories | Tests 


Erors and Failures | Tests Not Run | Text Output 


FIGURE 38.4 — Interface de NUnit 


La première chose à faire est de créer un nouveau projet (voir la figure 38.5). 
Appelez-le ProjetTest par exemple. Il faut ensuite ajouter une assembly de test ; allez 
dans Project > Add Assembly comme indiqué à la figure 38.6. 


Pour finir, allez pointer l’assembly de tests, à savoir MathTests.Unit.d11. NUnit ana- 
lyse l’assembly et fait apparaître la liste des tests qui composent notre assembly, en se 
basant sur les attributs TestFixture et Test (voir la figure 38.7). 

Nous pouvons à présent lancer les tests en cliquant sur Run! On s’aperçoit rapidement, 
en observant la figure 38.8, qu’il y a un test qui passe (icône verte) et un test qui échoue 
(icône rouge). 

Forcément, notre test n’était pas bon, il faut le réécrire. Nous voyons également qu’il 
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Reload Project Ctri+L 
Reload Tests Ctri+R 
Select Runtime 


Recent Projects 
Bt nn) 
ACTEURS 


FIGURE 38.5 — Création d’un nouveau projet NUnit 


FIGURE 38.6 — Ajout de l’assembly contenant les tests 
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[Categories | Tests 1m 


Ready [Test Cases:2 | | 


View Project Test Tools Help 


nents\Nunit 


s\Visual 2010\Projects 


ë 


Stop PCiDocuments\NunifProjet 


È Factorielle_AvecValeur10_Retoume1 
il Factorielle_AvecValeur3_Retoume6 


Erors and Failures | Tests Not Run | Text Output 


FIGURE 38.7 — Les tests présents dans l’assembly 


ile View Project Test Tools Help 


3 E-JK C:\Users\Nico-PC\Documents\Nunit\ProjetTest nunit 
F 3 C:\Users\Nico-PC\Documents\Nunit\..\Visual Studio 2010 
8 
o 
Æ ; 
6] JR Factorele_AvecValeur10_Retoume 1 rasseg: | ralleg: 1 Errors-U INCONCIUSIVE: U INVaNT U 
LEA :.@ Factorele_AvecValeur3_RetoumeS innared-1 Shinned-N Time À GRANT 
MathTests.Unit.MathTests.Factorielle AvecValeur10 R 
etournel: 
La valeur doit être égale à 1 
Expected: 1 


But was: 3628800 


CA Per , 


= 
MathTests.Unit MathTests.Factorielle AvecValeurl 
0_Retournel() dans C:\Users\Nico-PC\Documents 
AVisual Studio 2010\Projects 
\MaBibliothequeATester\MathTests.Unit 
\MathTests.cs:ligne 25 


Erors and Falures | Tests Not Run | Text Output 
Completed [Test Cases:2 (Tests Run:2 [Emos:0 [Faiures:1 |Time:49560547 » 


FIGURE 38.8 — Première exécution des tests 
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nous indique que le résultat attendu était 1 alors que le résultat obtenu est de 3628800. 
Nous pouvons également voir le message que nous avons demandé d’afficher en cas 
d'erreur. Le souci avec NUnit, c’est qu’à partir du moment où il à chargé la d11 pour 
lancer les tests, il n’est plus possible de faire de modifications, car toute tentative de 
compilation provoquera une erreur où il sera mentionné qu'il ne peut pas faire de modi- 
fications car le fichier est déjà utilisé ailleurs. Ce qui est vrai. Nous serons donc obligés 
de fermer puis de rouvrir NUnit. À noter que dans les versions payantes de Visual 
Studio, nous avons la possibilité de configurer NUnit en tant qu’outil externe, ce que 
nous ne pouvons pas faire avec la version gratuite. Il va falloir faire avec... C’est un 
des inconvénients de la gratuité. ..! Nous pouvons cependant un peu tricher en définis- 
sant un événement de post-compilation, qui consiste à lancer NUnit automatiquement. 
Pour cela, allez dans les propriétés du projet, onglet Événements de build et tapez la 
commande suivante : "C:\Program Files\NUnit 2.5.10\bin\net-2.0\nunit.exe" 
$(TargetFileName). Ici, nous indiquons qu'après la compilation, il va lancer le pro- 
gramme nunit.exe en prenant en paramètre le résultat de la compilation, représenté 
par la variable interne de Visual C# Express : « $(TargetFileName) » (voir la figure 
38.9). 


L#| MaBibliothequeATester - Microsoft Visual C# 2010 Express 
Fichier Edition Affichage Projet Déboguer Données Outils Fenêtre 2 
iddJ-Sd4| 389 -@-|>|æ#| 


MathTests.cs Math.cs 


4 


|A > 51 à - 


Application 


Ligne de commande de l'événement pré-build : 
Générer . 
Événements de build 


À « 0 
Déboguer 


Ressources Modifier pré-build … 


Ligne de commande de l'événement post-build : 


saauuop ap s221n0$ gi sjqno e 3109 .X 


Paramètres 


Chemins d'accès des références 


Signature 4 Gi LE 


Modifier post-build … 


FIGURE 38.9 - Modification des éléments de post-compilation 


Par contre, cela veut dire que NUnit va se lancer à chaque compilation, ce qui n’est peut- 
être pas le but recherché... Il faudra également fermer NUnit avant de pouvoir faire 
quoi que ce soit d’autre. À noter que maintenant que nous savons faire de l’introspection 
sur les méthodes et les attributs d’une classe, nous devrions être capables de créer une 
petite application qui exécute les tests automatiquement ! Pour en finir avec NUnit, 
notons qu’il y a beaucoup de méthodes permettant de vérifier si un résultat est correct. 
Regardons les assertions suivantes : 


1 | bool b = true; 
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Assert.IsTrue(b); 
string s = null; 
Assert.IsNull(s); 

int i = 10; 
Assert.Greater(i, 6); 


A Où À ND 


Elles parlent d’elles-mêmes. La première permet de vérifier qu’une condition est vraie. 
La deuxième permet de vérifier la nullité d’une variable. La dernière permet de vé- 
rifier qu’une variable est bien supérieure à une autre. À noter qu’elles ont chacune 
leur pendant (IsFalse, IsNotNull, Less). En regardant la complétion automatique, 
vous découvrirez d’autres méthodes de vérification, mais celles-ci sont globalement suf- 
fisantes. Nous pouvons également utiliser une syntaxe un peu plus parlante, surtout 
pour les anglophones : 


1 | Assert.That(i, Is.EqualTo(10)); 


Il est également possible d'utiliser un attribut pour vérifier qu’une méthode lève bien 
une exception, par exemple : 


[Test] 
[ExpectedException(typeof (FormatException))] 
public void Tolnt32_AvecChaineNonNumerique_LeveUneException() 
{ 
Convert.ToIlnt32("abc"); 


Oo Où À À D 


} 


Dans ce cas, le test passe si la méthode lève bien une FormatException. 


Avant de terminer, présentons deux attributs supplémentaires : les attributs SetUp et 
TearDown. Ils permettent de décorer des méthodes qui seront appelées respectivement 
avant et après chaque test. C’est l’endroit idéal pour factoriser des initialisations ou 
des nettoyages dont dépendent tous les tests. 


1| [TestFixture] 

2| public class MathTests 

3| { 

4 [SetUp] 

5 public void InitialisationDesTests() 
6 { 

7 // rajouter les initialisations 
8 } 

9 

10 [Test] 

11 public void Factorielle_AvecValeur3_Retourne6() 
12 { 

13 // test à faire 

14 } 

15 

16 [TearDown] 

17 public void NettoyageDesTests() 

18 { 
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19 // nettoyer les variables, 


Il existe plein d’autres choses utiles à dire sur NUnit, comme la description des autres 
attributs, mais il n’est pas utile de les voir toutes à notre niveau. N'hésitez pas à aller 
voir sur internet des informations plus poussées si vous ressentez le besoin d’approfondir 
votre maîtrise des tests. 


Le framework de simulacre 


Un framework de simulacre fournit un moyen de tester une méthode en l’isolant du 
reste du système. Imaginons par exemple une méthode qui permette de récupérer la 
météo du jour, en allant la lire dans une base de données. Nous avons ici un problème, 
car lorsque nous exécutons le test le lundi, il pleut. Quand nous exécutons le test le 
mardi, il fait beau, etc. Nous avons ici une information qui varie au cours du temps. Il 
est donc difficile de tester automatiquement que la méthode arrive bien à construire la 
météo du jour à partir de ces informations, vu qu’elles varient. Le but de ces frameworks 
est de pouvoir bouchonner le code dont notre développement dépend, afin de pouvoir 
le tester unitairement, sans dépendance et isolé du reste du système. Cela veut dire 
que dans notre test, nous allons remplacer la lecture en base de données par une fausse 
méthode qui renvoie toujours qu'il fait beau. Cependant, ceci doit se faire sans modifier 
notre application, sinon cela n’a pas d'intérêt. Voilà à quoi servent ces framework de 
simulacres. Il en existe plusieurs, plus où moins complexes. Citons par exemple Moq 
(prononcez « moque-you ») ou encore Moles (il y en a plein d’autres). L'intérêt de 
Mogq est qu'il est simple d’accès, nous allons le présenter rapidement. Il permet de faire 
des choses simples et facilement. Tandis que Moles est un peu plus évolué mais plus 
complexe à prendre en main. Vous y reviendrez sans doute ultérieurement. Pour le 
télécharger, utilisez le code web suivant : 


Télécharger Moq ) 


Code web : 225970 


Pas de système d'installation évolué, il y aura juste une d11 à référencer. Ajoutez donc 
la référence à la d11 moq.d1l qui se trouve dans le sous-répertoire NET40. Ensuite, 
pour pouvoir bouchonner facilement une classe, elle doit implémenter une interface. 
Imaginons la classe d'accès aux données suivante : 


1| public class Dal : IDal 

2| € 

3 public Meteo UbtenirLaMeteoDuJour() 

4 { 

5 // ici, c'est le code pour lire en base de données 

6 // mais finalement, peu importe ce qu'on y met vu qu'on 
va bouchonner la méthode 

7 throw new NotIlmplementedException(); 

8 } 

o[} 
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Qui implémente l'interface suivante : 


1| public interface IDal 

21 { 

3 Meteo ObtenirLaMeteoDuJour(); 
al} 


Avec l’objet Meteo suivant : 


1| public class Meteo 

2| € 

3 public double Temperature { get; set; } 
4 public Temps Temps { get; set; } 

5| } 


Et l’énumération Temps suivante : 


1| public enum Temps 
2| { 

3 Soleil, 

4 Pluie 

5| } 


Comme nous allons le voir, nous pouvons également écrire un test qui bouchonne 
l’appel à la méthode ObtenirLaMeteoDuJour, qui doit normalement aller lire en base 
de données, pour renvoyer un objet à la place. Pour bien montrer ce fonctionnement, 
j'ai fait en sorte que la méthode lève une exception, comme ça, si on passe dedans ça 


sera tout de suite visible. La méthode de test classique devrait être : 


1] [Test] 

2| public void ObtenirLaMeteoDuJour_AvecUnBouchon_RetourneSoleil() 
3| { 

4 IDal dal = new Dal); 

5 Meteo meteoDuJour = dal.0btenirLaMeteoDuJour(); 

6 Assert.AreEqual(25, meteoDuJour.Temperature); 

7 Assert.AreEqual (Temps.Soleil, meteoDuJour.Temps); 

8| } 


Si nous exécutons le test, nous aurons une exception. Utilisons maintenant Moq pour 


bouchonner cet appel et le remplacer par ce que l’on veut : 


1] [Test] 

2| public void ObtenirLaMeteoDuJour_AvecUnBouchon_RetourneSoleil() 
3| 

4 Meteo fausseMeteo = new Meteo { Temperature = 25, Temps = 


Temps.Soleil }; 
Mock<IDal> mock = new Mock<IDal>(); 
6 mock.Setup(dal => dal.0btenirLaMeteoDuJour ()).Returns( 
fausseMeteo); 


ot 


IDal fausseDal = mock.0bject; 
Meteo meteoDuJour = fausseDal.0btenirLaMeteoDuJour); 
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10 Assert.AreEqual(25, meteoDuJour.Temperature); 
11 Assert.AreEqual (Temps.Soleil, meteoDuJour.Temps); 
1217} 


On utilise l’objet générique Mock pour créer un faux objet du type de notre interface. 
On utilise la méthode Setup à travers une expression lambda pour indiquer que la 
méthode ObtenirLaMeteoDuJour retournera en fait un faux objet météo. Cela se fait 
tout naturellement en utilisant la méthode Returns (). L'avantage de ces constructions 
est que la syntaxe parle d’elle-même à partir du moment où l’on connaît les expressions 
lambdas. On obtient ensuite une instance de notre objet grâce à la propriété Object 
et c’est ce faux objet que nous pourrons comparer à nos valeurs. Bien sûr, ici, ce test 
n’a pas grand intérêt. Mais il faut le voir à un niveau plus général. Imaginons que nous 
ayons besoin de tester la fonctionnalité qui met en forme cet objet météo récupéré de la 
base de données ou bien l’algorithme qui nous permet de faire des statistiques sur ces 
données météos... Là, nous sommes sûrs de pouvoir nous baser sur une valeur connue 
de la dépendance à la base de données. Cela permettra également de décliner tous les 
cas possibles en changeant la valeur du bouchon et de faire les tests les plus exhaustifs 
possibles. Nous pouvons faire la même chose avec les propriétés. Imaginons la classe 
suivante dont la propriété valeur retourne un nombre aléatoire : 


1| public interface IGenerateur 
2| € 

3 int Valeur { get; } 

al} 

5 

6| public class Generateur : IGenerateur 
7| T 

8 private Random r; 

9 public Generateur() 

10 { 

11 r = new Random); 

12 ; 

13 

14 public int Valeur 

15 { 

16 get 

17 À 

18 return r.Next(0O, 100); 
19 } 
20 } 
21| } 


Nous pourrions avoir besoin de bouchonner cette propriété pour qu’elle renvoie un 
nombre connu à l’avance. Cela se fera de la même façon : 


1| Mock<IGenerateur> mock = new Mock<IGenerateur >() ; 

2| mock.Setup(generateur => generateur.Valeur).Returns (5); 
3 

4| Assert.AreEqual(5, mock.0Ubject.Valeur) ; 
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Ici, la propriété Valeur renverra toujours 5 en se moquant bien du générateur de 
nombre aléatoire... Je m’arrête là pour l’aperçu de ce framework de simulacre. Nous 
avons pu voir qu'il pouvait facilement bouchonner des dépendances nous permettant 
de faciliter la mise en place de nos tests unitaires. Rappelez-vous que pour qu’un test 
soit efficace, il doit pouvoir se concentrer sur un point précis du code sans être gêné 
par les dépendances éventuelles qui peuvent agir sur l’état du test à un instant « t ». 


En résumé 


— Les tests unitaires sont un moyen efficace de tester des bouts de code dans une 
application afin de garantir son bon fonctionnement. 

— Ils sont un filet de sécurité permettant de faire des opérations de maintenance, de 
refactoring ou d'optimisation sur le code. 

— Les frameworks de tests unitaires sont en général accompagnés d'outils permettant 
de superviser le bon déroulement des tests et la couverture de tests. 


482 


| 


Les types d'applications pouvant être 
développées en C 


Difficulté : 1 


ous savez quoi ? Avec le C# on peut créer autre chose que des applications console | 

On peut faire des applications avec des boutons et des menus, ou des sites web et 

même des jeux! Dans ce chapitre, je vais vous indiquer rapidement les différents 
types d'applications qu'on peut développer avec le C#. Chaque paragraphe de ce chapitre 
nécessiterait un livre entier pour être correctement traité, aussi, ce ne sera qu'un très 
bref aperçu. J'espère que vous me pardonnerez d'aller si vite mais je souhaite néanmoins 
piquer votre curiosité pour vous donner envie d'aller explorer tous ces nouveaux mondes 
qui s'ouvrent à vous! Il y aura certainement des notions que vous ne comprendrez pas 
complètement en fonction des thèmes abordés. Il faudra aller jeter un œil complémentaire 
sur internet ou attendre un prochain livre! 
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Créer une application Windows avec WPF 


Les applications Windows sont ce qu’on appelle des applications clients lourds. Elles 
s’exécutent directement depuis notre système d’exploitation. Nous pouvons créer toutes 
sortes d'applications, comme un traitement de texte, une calculatrice, etc. 


Nous avons déjà créé une application Windows à travers notre projet console, sauf que 
nous étions rapidement limités. Avec WPF, nous allons pouvoir créer des applications 
graphiques avec des boutons, des menus, etc. Bref, tout ce qui compose une application 
habituelle. WPF signifie Windows Presentation Foundation. 1 s’agit d’une bibliothèque 
permettant de réaliser des applications graphiques. Ces applications sont dites événe- 
mentielles car elles réagissent à des événements (clic sur un bouton, redimensionnement 
de la fenêtre, saisie de texte, etc.) 


Pour construire des applications WPF, nous aurons besoin de deux langages. Un lan- 
gage de présentation qui va permettre de décrire le contenu de notre fenêtre : le 
XAML(prononcez « xamelle ») et du C;} qui va permettre de faire tout le code métier. 


Créer une application web avec ASP.NET 


ASP.NET c’est la plateforme de Microsoft pour réaliser des applications web. C’est un 
peu comme PHP, sauf que, vous vous en doutez, ASP.NET s’appuie massivement sur 
le framework .NET. Et tout comme WPF, il s’agit de bibliothèques qui vont permettre 
de réaliser facilement son site web. 


Il existe deux ASP.NET : l'ASP.NET WebForms et l'ASP.NET MVC. ASP.NET 
WebForms c’est tout un mécanisme qui permet de faciliter la création d’une application 
web en faisant comme si c'était une application Windows. C'est-à-dire que le frame- 
work s'occupe de gérer toute la persistance d’informations entre les différents états 
des pages. Il permet aussi de travailler avec une approche événementielle, comme une 
application Windows. Le premier but d'ASP.NET WebForms était de faire en sorte 
que les personnes qui avaient déjà fait du développement Windows (avec des langages 
comme le Visual Basic ou autre) puissent facilement faire du développement web, dans 
un contexte qui leur serait familier. 


ASP.NET MVC est plus récent et offre une approche où le développeur doit bien 
connaître tous les mécanismes du web. Il offre également une plus grande maîtrise sur 
le rendu du site web. Enfin, il intègre par défaut tous les mécanismes éprouvés du fa- 
meux patron de conception (design pattern) MVC. On ne peut pas dire qu'ASP.NET 
WebForms soit mieux ou moins bien qu'ASP.NET MVC. Il s’agit de deux façons diffé- 
rentes de créer des sites web. Chacune a ses avantages et ses inconvénients. Par contre, 
les deux se basent sur un socle commun qui est le cœur d'ASP.NET. 
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Créer une application client riche avec Silverlight 


Nous avons vu qu’il existait des applications type clients lourds, comme les applications 
console ou les applications WPF. Nous avons également vu des applications web, avec 
ASP. NET WebForms ou ASP.NET MVC. Il existe quelque chose entre les deux, ce 
sont les applications dites « client riche ». Ce sont des applications qui ressemblent 
à des applications lourdes, mais qui s’exécutent à l’intérieur d’un navigateur internet 
plutôt que directement au niveau du système d’exploitation. Vous connaissez sûrement 
le célèbre « flash », très populaire grâce à la multitude de jeux disponibles sur internet. 
Microsoft possède également des bibliothèques permettant de réaliser des applications 
clients riches : Silverlight. 


Une application client riche s'exécute donc directement dans un navigateur internet, 
comme {nternet Explorer, Firefox ou Chrome. Ces applications s’exécutent dans un 
plugin du navigateur. Pour exécuter des applications flash ou des applications Silver- 
light, le navigateur devra posséder le plugin adéquat. Du fait qu’elles s’exécutent dans 
un navigateur, ces applications ont quelques restrictions. Elles ne peuvent par défaut 
pas accéder au contenu du disque dur de l'utilisateur, ce qui est finalement plutôt pas 
mal pour une application disponible directement sur internet. Elles s’exécutent unique- 
ment dans la zone mémoire du navigateur, une espèce de bac à sable dont on ne peut 
pas s'échapper et d’où il est impossible d'accéder aux ressources directes de l’ordina- 
teur sur lequel s’exécute l’application, au contraire des applications WPF par exemple. 
Ces applications clients riches ressemblent énormément aux applications clients lourds 
avec quelques restrictions. Silverlight est donc une espèce de WPF allégé qui ne garde 
que l'essentiel de l’essentiel. Nous serons donc capables de réaliser des applications 
s’exécutant dans notre navigateur grâce au C# et au XAML. 


Le graphisme et les jeux avec XNA 


Eh oui, il est aussi possible de réaliser des jeux avec le C#. Même si tout ce qu’on a 
vu auparavant permet la réalisation d'applications de gestion, Microsoft dispose aussi 
de tout ce qu’il faut pour réaliser des jeux, grâce à XNA. XNA est un ensemble de 
bibliothèques permettant de créer des applications graphiques mais c’est également un 
outil permettant d'intégrer des contenus facilement dans des jeux (comme des images 
ou des musiques). Avec XNA, il est possible de faire des jeux qui fonctionnent sous 
Windows mais également sous Xbox ou sur les téléphones Windows. 


Créer une application mobile avec Windows Phone 7 


Le C# nous permet également de développer des applications pour téléphones mobiles 
équipés du système d’exploitation Windows Phone. C’est un point très important car 
il est possible de mutualiser beaucoup de choses que nous avons apprises pour les 
transposer dans le monde en plein essor des smartphones. 


Ce qui est intéressant avec les applications Windows Phone c’est que nous réutilisons 
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le savoir que nous avons pu acquérir dans les autres types d'applications C#. En effet, 
pour réaliser des applications de gestion, nous allons utiliser Silverlight et pour réaliser 
des jeux, nous utiliserons XNA. 


Créer un service web avec WCF 


Avec le C;£ il est également très facile de créer des services web. Un service web per- 
met en général d’accéder à des fonctionnalités depuis n’importe où, à travers internet. 
Citons par exemple les services web d'Amazon qui nous permettent de récupérer des in- 
formations sur des livres, ou encore des services web qui permettent d'obtenir la météo 
du jour. Bref, c’est un moyen de communication entre applications hétérogènes poten- 
tiellement situées à des emplacements physiques très éloignés. En imaginant que nous 
ayons également besoin d'exposer des méthodes à l’extérieur, pour qu’un fournisseur 
vienne consulter l’état de nos commandes ou qu’un client puisse suivre l’avancée de la 
sienne. .., nous allons devoir créer un service web. Le framework .NET dispose de tout 
un framework pour cela qui s'appelle WCF : Windows Communication Foundation. 


Un service web est une espèce d’application web qui répond à des requêtes permettant 
d'appeler une méthode avec des paramètres et de recevoir en réponse le retour de la 
méthode. L'intérêt d’un service web est qu’il est indépendant de la technologie. Même 
si on écrit un service web avec du C#, il peut être appelé par du java ou du PHP. 


Retrouvez des explications plus complètes et des exemples pratiques via le code web 
suivant : 


Aller plus loin 
Code web : 777901 
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Pour conclure 


Ca y est, ce livre est terminé. 


Le C# n’a (presque) plus de secret pour nous. Tout au long de ce livre nous avons 
découvert comment développer des applications avec le C#. Il à fallu dans un premier 
temps apprendre les bases du C#, ce qui n’est pas sans douleurs quand on n’a jamais 
fait de programmation! Pouvoir appréhender correctement ce qu’est une variable ou 
comment dérouler son premier programme n’est pas une mince affaire. Mais petit à 
petit, nous avons relevé le défi en présentant des fonctionnalités qui nous serviront tout 
le temps, pour la moindre petite application. 


Forts de ces nouveaux apprentissages, nous avons ensuite découvert le monde de la 
programmation orientée objet et comment le C# était un vrai langage orienté objet. À 
travers des notions comme les classes et autres propriétés, nous avons pu tirer parti de 
cette façon de modéliser des applications pour les adapter à nos besoins. Je le reconnais, 
ce n’est pas une partie facile. Les différents concepts peuvent donner rapidement mal 
à la tête! Ce qui est certain c’est que petit à petit, ils vont devenir de plus en plus 
clairs. On peut très bien commencer à développer en orienté objet sans vraiment en 
maîtriser toutes les subtilités. C’est en pratiquant et en restant curieux sur le sujet que 
vous progresserez. 


Enfin, nous avons poussé un peu plus loin dans les arcanes du C#, nous permettant 
de réaliser des applications de plus en plus compliquées. Nous avons également montré 
comment utiliser les bases de données avec Entity Framework. Savoir lire et écrire dans 
une base de données est un élément fondamental pour toute application de gestion qui 
se respecte. Nous avons également aperçu ce que l’on pouvait faire avec le C#. De 
l'application Windows, en passant par les jeux ou les applications web, le C# combiné 
au framework .NET permet vraiment de faire beaucoup de choses! 


Vous avez désormais les clés pour démarrer. Il ne reste plus qu’à vous plonger dans 
les domaines qui vous intéressent afin de réaliser les applications dont vous avez envie. 
Pourquoi pas une application pour les smartphones ? C’est très à la mode. 


N'oubliez pas que c’est à force de pratiquer, d’essayer de créer des petites applications 
de rien du tout, que vous finirez par être un développeur confirmé, capable à son tour 
d’aider les autres. N'hésitez pas à faire et à refaire les TP. Après avoir fini le livre, 
vous pourrez sûrement améliorer vos premières versions des TP. Créez-en, fixez vous 
des petits défis et venez les échanger avec nous sur le Site du Zéro! :;-) 


Index 


A 
AbSÉLACR. Lisa ueisd leu us ciuerss 233 
AGCION: ads de ss didenconade 327 
AFÉCLATIONLE 4 auras ad he een 38 
application 
MObilé;Err ns sise reihenc 485 
MED aies ea 484 
Windows ........................ 484 
ACTES re conceu ere mentonbasese 144 
ASPINET :: 23852542 ns seu eniuress 484 
assembly enr tennisies 8, 78, 361 
AMOEIDUÉ 24 22 needed doi De mure duo den 391 
auto-complétion....................... 31 
B 
base de données ................. 416, 432 
bibliothèque ......................... 361 
booléen ::: isa sdis.sasbisees due 47 
bouclé séisme sstitinertemessdhensess 91 
ÉOLL: Lire ed issue 92 
ÉOTEACR 15554 idduasassane made 95 
MODE 11:22:32 15 cuede dc daueda tee 98 
break. same earelraiarss 100 
C 
CATACLÈTÉ: La nb dridede en 121 
caractères SPÉCIAUX. ................... 41 
CASÉINE 4 ca stars es deniers 112 
chaîne de connexion ............. 403, 453 
CLASSE se detses den Laria Mdr eds 166 
clé étrangère: :24. siussussetasdesses 448 
Clé Primaire. saue. dose mes 448 
CLR de sus done se assassins 6 
COMMENtAITE .............,............ 31 


COMPATAÏSON...................... 46, 222 
COMDIATION.. 444 ass cruels 21 
complétion automatique............... 31 
concaténation ......................... 38 
condition.............................. 45 
CONMBUTATION 4 a8 8 entame aire dre 399 
console................................ 18 
CONSÉTUCHEUT + :.54 eus dome desde 184 
CONDINUE . 444 serum genes 101 
CONVÉTSION 3 cv5 bas entde éé dase sé e sas TL 
CUIEUTÉ : 34 La doper penis tasse 385 
D 
débogueur..hissssssssiseudubeneuess à 125 
décrémentation........................ 40 
délégué... ana teen 322 
E 
ÉLSE 5: 28 64888 germe ont sents 46 
ÉLSENLÉ Less inesser idee ere 49 
encapsulation ........................ 159 
ÉTITEPISTTÉMENL... ésasisantemorentres 432 
Entity Framework.................... 432 
ÉRUMÉTAUON 24 pee sh peaue Miyse died 379 
ÉVÉNL Eire tmaateese haben 330 
CXCEDUION 2 ss usines aieesesedese 336 
CAC 4e ture st eaidae sion 337 
CRTOW. SLI ISA EAN AIT rare ees 342 
EXTENSION 222 rte ouest 423 
F 
FINalLY so uucssses sos dressent 347 
ÉOT 4 Sirsbesdd aie das eme dose 92 
ÉOTEACR . « uuseue cesse nur 70 


488 


L0EMALAGE. 58 ei dents 384 
framework .NET ....................... 8 
FÜNC redemarre sa dede 327 
G 
garbage collector ..................... 379 
BÉNÉTIQUE 254 save dosage dre ds dia 282 
BLOUDES raider) ANA an 407 
H 
Héritagé.sisiesesiertmarive: 159, 194, 372 
I 
IDE: 22 MAR SM 12 
SE 46 
incrémentation ........................ 40 
INGÉTIAC 44 sr 5 dues eau de Là béta ve 225 
inbérnal;::eiiusisnmesdroneenpereate 368 
L 
DINQ ES esse aura tas 416 
LS TR 70 
M 
MÉMOIRO Ed ci Ro 373 
méthode .............................. 58 
Microsoft SQL Server ................ 444 
modélisation ......................... 433 
MOULO 4425-10 p MN ER den a 101 
MONO: an A ER AMAR ENTRE 12 
MO ess eee een Re en 2 479 
N 
 e 187 
NÉE RSR ie 471 
O 
ODJÉCb ei sessssunamnnrsusetersat 203 
ODÉTALION à sas Se ndentiete dose mn dde 38 
OU: enceinte made imac 269 
P 
partial: uuimmieursearalanres 236 
DIEU ,siiee either est hier 373 
pile des appels ....................... 132 
point d’arrêt......................... 127 
polymorphisme.................. 161, 211 
PriVAbe 4:05 dansante 172 


protected... situées 196 
PUBLIC à 4 res dues cher aude dou 172 
R 
ramasse-miettes...................... 375 
Te. shte- cet eme nie uen 267 
TéfÉ NC. 224 méme mens desneséas 193 
réflexion ............................. 393 
TÉQUÊLÉS Hadrien denied des 416 
S 
sealed ..:irevetarsetsurearessort 372 
Silverhiohtsssessmiesisatiesatius 485 
SO EP 416 
SABLE 120 nie dite ad ie 238 
SCEUCTUTÉ si ace Noos dedans ane 274 
SUbSGCUHION: 4 à 4: sésase cyber se 161, 207 
SULECNs dre sas ontiideneeeides 49 
SA ULTED CE ET 27 
T 
tableau sie rss ee 68 
TES serres ris eme inedeienr 466 
SMMIACTÉ 24 se dradee seine 479 
CRIS ini Me VIe eee tee 189 
Le] 
USINE nine aides 76 
V 
VAlEUT 2 42 due ja da au vase ea sugauhisus 193 
DA D 244 
variables 3er italien 35 
Visual C# 2010 Express .............. il 
W 
WCF huis hisisusn is agemausesestel 486 
WhÈTÉ..;ssésirimaiienaatisetetsdue 291 
0 0 EP 98 
Windows Phone 7.................... 485 
WP: Mises dar oserearesesermosa 484 
X 
XML ssissssisinmicnessmmetéers ane 398 
XNA ia dau nes smaen ondes 485 
d'a 
Yleldis: tite sue cuire 379 


489 


Dépôt légal : mars 2012 
ISBN : 978-2-9535278-6-5 
Code éditeur : 978-2-9535278 


Imprimé en France 


Achevé d'imprimer le 20 mars 2012 
sur les presses de Corlet Imprimeur (Condé-sur-Noireau) 
Numéro imprimeur : 144485 


RM y 
SE 


1 24104 
OS 

to ® 
?n ement 


Ÿ 
6, se 
LR e 

T'agit pout \ 


Mentions légales : 


Conception couverture : Fan Jiyong 


Illustrations chapitres : Fan Jiyong 


ER EN C# 


programmation ° ? 
s, découvrez pas à 
nez la programmation 


APPRENEZ À DÉVELOPP 


vous n'avez aucun 
e à ce livre conçu 
lez le fram 


e connaissance en 


pour les débutant 


is 
Vous désirez apprendre le C# ee 
Cet ouvrage est fait pour _. | or 
de Micro 
jangage phare Eu 
ie er et bien d’autres choses enc 
orien 


ework NET, appre 


Plus de 30 chapitres de difficulté progressive 


Des exercices réguliers sous forme de TP Pi Es mu 
Un livre entièrement en couleur al) 


are de tlécharger Micr059f 


À propos de l'auteur 


Nicolas Hilaire 


Passionné de développement depuis tout petit, 
Nicolas Hilaire est aujourd'hui ingénieur en 
informatique pour une grande entreprise 


\ d'e-commerce française 
4 


Suite à plusieurs publications sur le C++/CLI, il a obtenu en 2007 le 
tite de Microsoft Valuable Professional Visual C++. || partage 
depuis sa passion pour le C# et ASP.NET et obtient le titre 
Microsoft Valuable Professional ASP.NET en 2010. Curieux des 
autres langages, il continue à s'intéresser aux nouvelles 
technologies qui pointent le bout de leur nez 


Ce livre est issu du Site du Zéro 

Retrouvez dans ce livre les cours du Site du Zéro dans une 
édition revue et corrigée avec un nouveau chapitre inédit du 
même auteur ! 

Téléchargez les codes source en ligne grâce aux « codes web » 
inclus dans ce livre. 


TI ges M Simple nd 


9 782953 527865 


